直接在controller中@Autowired注入request对象问题

直接在controller中直接注入requesete和response对象有没有线程安全问题呢?

@Controller
@RequestMapping("test")
public class TestController {

    @Autowired   // 直接注入request
    private HttpServletRequest request;  

    @Autowired
    private HttpServletResponse response;

    public TestController() {
        System.out.println("创建TestController");//在这里打上断点,查看spring是怎样注入request和response对象的
    }
}

​ 首先,可以知道controller在spring容器中是单例存在的,那么如果一个请求还没完成之前,另外一个请求就过来了,那么会不会覆盖掉原来的request或response对象呢?

​ 在TestController的构造方法上打上断点,查看TestController这个bean的创建过程;populateBean方法在给TestConroller这个bean对象进行属性赋值,

在这里插入图片描述

​ 属性赋值完成后,可以看到:注入的其实都是一个代理

在这里插入图片描述

那么在运行的过程中,调用代理对象的方法将会先走代理对象的方法,

[猜想:代理对象将会从RequestContextHolder中的ThreadLocal本地线程安全的变量中拿到request对象,(在多线程的环境下,每个线程只会拿到属于当前线程的request对象),因此就不会出现线程安全问题]

也就是说这个问题其实就是:注入的是代理对象,而并不是每次请求来了,才注入。
[证明:在RequestContextHolder对象中的静态方法getRequestAttributes打上断点,]

public abstract class RequestContextHolder  {
    
    private static final ThreadLocal<RequestAttributes> 
        					requestAttributesHolder = new NamedThreadLocal<>("Request attributes");
    
    public static RequestAttributes getRequestAttributes() {
       RequestAttributes attributes = requestAttributesHolder.get(); // 在这里打上断点
       if (attributes == null) {
          attributes = inheritableRequestAttributesHolder.get();
       }
       return attributes;
    }
}

接下来访问controller的test02这个方法,我们可以看到,进入到了上面这个断点里面

@Controller
@RequestMapping("test")
public class TestController {

    @Autowired
    private HttpServletRequest request;

    @Autowired
    private HttpServletResponse response;

    public TestController() {
        System.out.println("创建TestController");
    }

    @RequestMapping("test02")  // 访问controller的这个方法
    @ResponseBody
    public String test02() throws InterruptedException {
        HttpSession session = request.getSession();
        return "ok";

    }
}

在这里插入图片描述

这下就都清楚了

plus:
最近又看了一遍spring,关键点就在于,DefaultListatbleBeanFactory的doResovleDependency这个方法里,顾名思义,这个方法就是用来解析依赖的,在spring的容器里为了能够给属性注入依赖,spring会保存一个resolveDependencies的依赖的集合,而spring在web扩展的包里面,往这个依赖的集合里面放了解析的依赖,因而就能够注入request对应的的ObjectFactory,然后在解析依赖的逻辑中,判断如果是个接口的话,那么就使用Proxy.newInstance方法创建jdk的动态代理。既然是jdk的动态代理,那么我们自然要去关注创建代理时传进去的InvocationHandler,而这个Handler就是图示标注的。重点看看这个InvocationHandler的invoke方法就能明白这是怎么一回事了。它就是利用ObjectFactory这个延迟获取对象,然后通过反射调用方法的。

  • 2
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,我可以为你提供一个简单的登录案例,使用Spring Boot框架和相关注解来实现。 首先,我们需要创建一个名为"UserController"的控制器类。该类将处理用户登录请求,并返回相应的响应。 ```java @Controller public class UserController { @Autowired private UserService userService; @PostMapping("/login") @ResponseBody public ResponseEntity<String> login(@RequestBody UserDto userDto) { User user = userService.login(userDto.getUsername(), userDto.getPassword()); if (user != null) { return new ResponseEntity<>("Login Successful", HttpStatus.OK); } else { return new ResponseEntity<>("Login Failed", HttpStatus.UNAUTHORIZED); } } } ``` 在上面的代码,我们使用了@Controller注解来标记该类为控制器类,@Autowired注入UserService实例,@PostMapping注解用于处理HTTP POST请求,@ResponseBody注解用于将响应体直接返回给客户端,而无需使用模型和视图。@RequestBody注解用于将请求体转换为UserDto对象,然后我们使用UserService实例进行用户登录验证,如果用户存在,则返回成功响应,否则返回失败响应。 接下来,我们需要创建一个UserService类,该类将处理用户登录逻辑。 ```java @Service public class UserService { @Autowired private UserRepository userRepository; public User login(String username, String password) { User user = userRepository.findByUsername(username); if (user != null && user.getPassword().equals(password)) { return user; } else { return null; } } } ``` 在上面的代码,我们使用了@Service注解来标记该类为服务类,@Autowired用于注入UserRepository实例,该类将处理用户登录逻辑。我们首先通过调用UserRepository的findByUsername方法来获取用户实例,然后比较用户密码是否与传递的密码相同。如果是,则返回用户实例,否则返回null。 最后,我们需要创建一个名为"UserDto"的数据传输对象类,该类将用于从请求体提取用户名和密码。 ```java @Data public class UserDto { private String username; private String password; } ``` 在上面的代码,我们使用了@Data注解来生成getter和setter方法,以及equals、hashCode和toString方法,这些方法将用于从请求体提取用户名和密码。 最后,我们需要创建一个名为"UserRepository"的接口,该接口将扩展JpaRepository接口,并提供自定义方法来查询用户实例。 ```java @Repository public interface UserRepository extends JpaRepository<User, Long> { User findByUsername(String username); } ``` 在上面的代码,我们使用了@Repository注解来标记该接口为存储库接口,该接口将扩展JpaRepository接口,并提供自定义方法findByUsername来查询用户实例。 以上就是使用Spring Boot框架和相关注解实现登录案例的全部内容。希望可以对你有所帮助。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值