SpringMvc中获取Request

目录

 ●  Controller中加参数
 ●  自动注入
 ●  基类中自动注入
 ●  手动调用

 ●  @ModelAttribute方法

Controller中加参数


  1. @Controller

  2. public class TestController {

  3. @RequestMapping("/test")

  4. public void test(HttpServletRequest request) {

  5. ......

  6. }

  7. }

Controller中获取request对象后,如果要在其他方法中(如service方法、工具类方法等)使用request对象,需要在调用这些方法时将request对象作为参数传入

此时request对象是方法参数,相当于局部变量,毫无疑问是线程安全的。

自动注入


  1. @Controller

  2. public class TestController{

  3. @Autowired

  4. private HttpServletRequest request; //自动注入request

  5. @RequestMapping("/test")

  6. public void test() throws InterruptedException{

  7. ......

  8. }

  9. }

使用这种方式,当Bean(本例的TestController)初始化时,Spring并没有注入一个request对象,而是注入了一个代理(proxy);当Bean中需要使用request对象时,通过该代理获取request对象。request实际上是一个代理:代理的实现参见AutowireUtils的内部类ObjectFactoryDelegatingInvocationHandler。

调用request的方法method时,实际上是调用了由objectFactory.getObject()生成的对象的method方法;objectFactory.getObject()生成的对象才是真正的request对象。

objectFactory的类型为WebApplicationContextUtils的内部类RequestObjectFactory;而RequestObjectFactory要获得request对象需要先调用currentRequestAttributes()方法获得RequestAttributes对象,生成RequestAttributes对象的核心代码在类RequestContextHolder中,生成的RequestAttributes对象是线程局部变量(ThreadLocal),因此request对象也是线程局部变量;这就保证了request对象的线程安全性。

基类中自动注入


  1. public class BaseController {

  2. @Autowired

  3. protected HttpServletRequest request;

  4. }

  5. @Controller

  6. public class TestController extends BaseController {

  7. }

与方法2相比,避免了在不同的Controller中重复注入request;但是考虑到java只允许继承一个基类,所以如果Controller需要继承其他类时,该方法便不再好用。

手动调用


  1. @Controller

  2. public class TestController {

  3. @RequestMapping("/test")

  4. public void test() throws InterruptedException {

  5. HttpServletRequest request = ((ServletRequestAttributes) (RequestContextHolder.currentRequestAttributes())).getRequest();

  6. .......

  7. }

  8. }

通过自动注入实现与通过手动方法调用实现原理差不多。因此本方法也是线程安全的。

优点:可以在非Bean中直接获取。缺点:如果使用的地方较多,代码非常繁琐;因此可以与其他方法配合使用。

@ModelAttribute方法


  1. @Controller

  2. public class TestController {

  3. private HttpServletRequest request; 此处线程不安全

  4. @ModelAttribute

  5. public void bindRequest(HttpServletRequest request) {

  6. this.request = request; 此处request线程安全

  7. }

  8. @RequestMapping("/test")

  9. public void test() throws InterruptedException {

  10. ......

  11. }

  12. }

@ModelAttribute注解用在Controller中修饰方法时,其作用是Controller中的每个@RequestMapping方法执行前,该方法都会执行。bindRequest()的作用是在test()执行前为request对象赋值。虽然bindRequest()中的参数request本身是线程安全的,但由于TestController是单例的,request作为TestController的一个域,无法保证线程安全。


原文发布时间为:2018-11-16
本文来自云栖社区合作伙伴“Java杂记”,了解相关信息可以关注“Java杂记”。
上一篇:使用HeartBeat实现高可用HA的配置过程详解


下一篇:卷影副本--恢复文件到以前的版本