【Spring】 Bean 注入 HttpServletRequest 能保证线程安全的原理

文章目录

    • 前言
    • [1. 图示](#1. 图示)
    • [2. 源码坐标](#2. 源码坐标)
    • 后记

前言

今天看了一段老业务代码,HttpServletRequest 被注入后直接用于业务逻辑。

好奇Spring是如何解决线程安全问题。

java 复制代码
@Controller
public class TestController {

    @Resource
    HttpServletRequest request;
    
    @ResponseBody
    @GetMapping("/test")
    public String test() {
        return request.getQueryString();
    }
}

本质问题是 Spring 如何装配 Controller 中 HttpServletRequest 的bean依赖

1. 图示

2. 源码坐标

可以在代码层面打断点验证,版本:SpringBoot 2.7.5 (自行集成WebStarter)

  • 加载web相关的bean定义
java 复制代码
ServletWebServerApplicationContext.java:141
  • 设置工厂
java 复制代码
WebApplicationContextUtils#registerWebApplicationScopes(org.springframework.beans.factory.config.ConfigurableListableBeanFactory, javax.servlet.ServletContext)
  • 具体的工厂方法定义(定义是从线程中取request)
java 复制代码
RequestObjectFactory#getObject
java 复制代码
// 关键代码块
@Override
public ServletRequest getObject() {
	return currentRequestAttributes().getRequest();
}
  • 启动时解析依赖,将工厂方法植入
java 复制代码
org.springframework.beans.factory.support.DefaultListableBeanFactory#doResolveDependency
org.springframework.beans.factory.support.DefaultListableBeanFactory#findAutowireCandidates
org.springframework.beans.factory.support.AutowireUtils#resolveAutowiringValue
  • 接收请求
java 复制代码
FrameworkServlet.java:1003  将响应放在ThreadLocal中
java 复制代码
// 关键代码块
if (requestAttributes != null) {
	RequestContextHolder.setRequestAttributes(requestAttributes, this.threadContextInheritable);
}
  • 回调工厂方法创建 Request
java 复制代码
AutowireUtils.java:292
java 复制代码
// 关键代码
return method.invoke(this.objectFactory.getObject(), args)

后记

  • 解决 HttpServletRequest 的线程安全问题的思路
    • 工厂方法定义, 将 Request 的创建约束为从 ThreadLocal 中取。
    • 收到请求 ,将 Request 放入ThreadLocal 中。(请求时设置 Request)
    • Controller 实例化,使用动态代理技术回调工厂方法。(业务逻辑消费 Request)

  • 建议按官方建议替换成方法参数,语义更加清晰。
java 复制代码
    @ResponseBody
    @GetMapping("/test")
    public String test(HttpServletRequest request) {
        return request.getQueryString();
    }
相关推荐
用户83071968408224 分钟前
Spring Boot 项目中日期处理的最佳实践
java·spring boot
JavaGuide1 小时前
Claude Opus 4.6 真的用不起了!我换成了国产 M2.5,实测真香!!
java·spring·ai·claude code
IT探险家1 小时前
Java 基本数据类型:8 种原始类型 + 数组 + 6 个新手必踩的坑
java
花花无缺1 小时前
搞懂new 关键字(构造函数)和 .builder() 模式(建造者模式)创建对象
java
用户908324602731 小时前
Spring Boot + MyBatis-Plus 多租户实战:从数据隔离到权限控制的完整方案
java·后端
桦说编程2 小时前
实战分析 ConcurrentHashMap.computeIfAbsent 的锁冲突问题
java·后端·性能优化
玹外之音3 小时前
Spring AI MCP 实战:将你的服务升级为 AI 可调用的智能工具
spring·ai编程
来一斤小鲜肉4 小时前
Spring AI入门:第一个AI应用跑起来
spring·ai编程
NE_STOP6 小时前
springMVC-常见视图组件与RESTFul编程风格
spring
程序员清风6 小时前
用了三年AI,我总结出高效使用AI的3个习惯!
java·后端·面试