RequestContextHolder详解

最近遇到的问题是在service获取request和response,正常来说在service层是没有request的,然而直接从controlller传过来的话解决方法太粗暴,后来发现了SpringMVC提供的RequestContextHolder遂去分析一番,并借此对SpringMVC的结构深入了解一下,后面会再发文章详细分析源码

1.RequestContextHolder的使用

RequestContextHolder顾名思义,持有上下文的Request容器.使用是很简单的,具体使用如下:

复制代码

 
  1. //两个方法在没有使用JSF的项目中是没有区别的

  2. RequestAttributes requestAttributes = RequestContextHolder.currentRequestAttributes();

  3. //RequestContextHolder.getRequestAttributes();

  4. //从session里面获取对应的值

  5. String str = (String) requestAttributes.getAttribute("name",RequestAttributes.SCOPE_SESSION);

  6. HttpServletRequest request = ((ServletRequestAttributes)requestAttributes).getRequest();

  7. HttpServletResponse response = ((ServletRequestAttributes)requestAttributes).getResponse();

复制代码

看到这一般都会想到几个问题:

  1. request和response怎么和当前请求挂钩?
  2. request和response等是什么时候设置进去的?

2.解决疑问

2.1 request和response怎么和当前请求挂钩?

首先分析RequestContextHolder这个类,里面有两个ThreadLocal保存当前线程下的request,关于ThreadLocal可以参考我的另一篇博文[Java学习记录--ThreadLocal使用案例]

复制代码

 
  1. //得到存储进去的request

  2. private static final ThreadLocal<RequestAttributes> requestAttributesHolder =

  3. new NamedThreadLocal<RequestAttributes>("Request attributes");

  4. //可被子线程继承的request

  5. private static final ThreadLocal<RequestAttributes> inheritableRequestAttributesHolder =

  6. new NamedInheritableThreadLocal<RequestAttributes>("Request context");

复制代码

再看`getRequestAttributes()`方法,相当于直接获取ThreadLocal里面的值,这样就保证了每一次获取到的Request是该请求的request.

复制代码

 
  1. public static RequestAttributes getRequestAttributes() {

  2. RequestAttributes attributes = requestAttributesHolder.get();

  3. if (attributes == null) {

  4. attributes = inheritableRequestAttributesHolder.get();

  5. }

  6. return attributes;

  7. }

复制代码

2.2request和response等是什么时候设置进去的?

找这个的话需要对springMVC结构的`DispatcherServlet`的结构有一定了解才能准确的定位该去哪里找相关代码.

在IDEA中会显示如下的继承关系.

左边1这里是Servlet的接口和实现类.

右边2这里是使得SpringMVC具有Spring的一些环境变量和Spring容器.类似的XXXAware接口就是对该类提供Spring感知,简单来说就是如果想使用Spring的XXXX就要实现XXXAware,spring会把需要的东西传送过来.

那么剩下要分析的的就是三个类,简单看下源码

1. HttpServletBean 进行初始化工作

2. FrameworkServlet 初始化 WebApplicationContext,并提供service方法预处理请

3. DispatcherServlet 具体分发处理.

那么就可以在FrameworkServlet查看到该类重写了service(),doGet(),doPost()...等方法,这些实现里面都有一个预处理方法`processRequest(request, response);`,所以定位到了我们要找的位置

查看`processRequest(request, response);`的实现,具体可以分为三步:

  1. 获取上一个请求的参数
  2. 重新建立新的参数
  3. 设置到XXContextHolder
  4. 父类的service()处理请求
  5. 恢复request
  6. 发布事

复制代码

 
  1. protected final void processRequest(HttpServletRequest request, HttpServletResponse response)

  2. throws ServletException, IOException {

  3. long startTime = System.currentTimeMillis();

  4. Throwable failureCause = null;

  5. //获取上一个请求保存的LocaleContext

  6. LocaleContext previousLocaleContext = LocaleContextHolder.getLocaleContext();

  7. //建立新的LocaleContext

  8. LocaleContext localeContext = buildLocaleContext(request);

  9. //获取上一个请求保存的RequestAttributes

  10. RequestAttributes previousAttributes = RequestContextHolder.getRequestAttributes();

  11. //建立新的RequestAttributes

  12. ServletRequestAttributes requestAttributes = buildRequestAttributes(request,

  13. response, previousAttributes);

  14. WebAsyncManager asyncManager = WebAsyncUtils.getAsyncManager(request);

  15. asyncManager.registerCallableInterceptor(FrameworkServlet.class.getName(),

  16. new RequestBindingInterceptor());

  17. //具体设置的方法

  18. initContextHolders(request, localeContext, requestAttributes);

  19. try {

  20. doService(request, response);

  21. }

  22. catch (ServletException ex) {

  23. failureCause = ex;

  24. throw ex;

  25. }

  26. catch (IOException ex) {

  27. failureCause = ex;

  28. throw ex;

  29. }

  30. catch (Throwable ex) {

  31. failureCause = ex;

  32. throw new NestedServletException("Request processing failed", ex);

  33. }

  34. finally {

  35. //恢复

  36. resetContextHolders(request, previousLocaleContext, previousAttributes);

  37. if (requestAttributes != null) {

  38. requestAttributes.requestCompleted();

  39. }

  40. if (logger.isDebugEnabled()) {

  41. if (failureCause != null) {

  42. this.logger.debug("Could not complete request", failureCause);

  43. }

  44. else {

  45. if (asyncManager.isConcurrentHandlingStarted()) {

  46. logger.debug("Leaving response open for concurrent processing");

  47. }

  48. else {

  49. this.logger.debug("Successfully completed request");

  50. }

  51. }

  52. }

  53. //发布事件

  54. publishRequestHandledEvent(request, response, startTime, failureCause);

  55. }

  56. }

复制代码

再看initContextHolders(request, localeContext, requestAttributes)方法,把新的RequestAttributes设置进LocalThread,实际上保存的类型为ServletRequestAttributes,这也是为什么在使用的时候可以把RequestAttributes强转为ServletRequestAttributes.

复制代码

 
  1. private void initContextHolders(HttpServletRequest request,

  2. LocaleContext localeContext,

  3. RequestAttributes requestAttributes) {

  4. if (localeContext != null) {

  5. LocaleContextHolder.setLocaleContext(localeContext,

  6. this.threadContextInheritable);

  7. }

  8. if (requestAttributes != null) {

  9. RequestContextHolder.setRequestAttributes(requestAttributes,

  10. this.threadContextInheritable);

  11. }

  12. if (logger.isTraceEnabled()) {

  13. logger.trace("Bound request context to thread: " + request);

  14. }

  15. }

复制代码

因此RequestContextHolder里面最终保存的为ServletRequestAttributes,这个类相比`RequestAttributes`方法是多了很多.

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/178566.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

开启核磁数据处理新篇章-MestReNova(MNOVA14)助您轻松解读科学界密码

在科学研究领域&#xff0c;核磁共振&#xff08;NMR&#xff09;技术被广泛应用于分析和解读化学物质的结构和性质。而MestReNova&#xff08;MNOVA14&#xff09;作为一款专业的核磁数据处理软件&#xff0c;凭借其强大的功能和易用性&#xff0c;已成为众多科研人员的首选工…

基于单片机的公交车报站系统(论文+源码)

1系统设计 本次课题为基于单片机的公交车报站系统&#xff0c;在此主要是基于Proteus平台展开设计&#xff0c;因此结合Proteus平台的特性&#xff0c;将功能设计如下&#xff1a; &#xff08;1&#xff09;公交车具有上行和下行两种状态&#xff0c;可以通过按键进行手动播…

spring cloud之服务熔断

四、Hystrix组件(*) - 官网&#xff1a;https://github.com/Netflix/Hystrix - 在分布式环境中&#xff0c;许多服务依赖项不可避免的会失败。Hystrix是一个库&#xff0c;它通过添加延迟容忍和容错逻辑来帮助您控制这些分布式服务之间的交互。Hystrix通过隔离服务之间的访问点…

Python OpenCV 通过trackbar调整图像亮度对比度颜色

上一篇文章通过设置固定值的方式来调整图像&#xff0c;这篇文章通过trackbar来动态调整参数&#xff0c;从而实时展现图像处理结果&#xff0c;得到想要的图像处理参数。 1. 创建trackbar import cv2 import numpy as npdef nothing(x):passcv2.namedWindow(image) # 创建5个…

Postman工具简介

介绍 Postman是一个商业的接口测试工具。免费的版本也可以使用不少功能。 官网&#xff1a;https://www.postman.com/ 下载、安装、应用界面 下载 安装、安装成功以后的应用界面 双击下载下来的可执行文件进行安装&#xff0c;出现如下界面&#xff1a; 可以注册一个账…

如果使用Vue要做根据已有的图形填入到指定的单元格中,你会怎么做?

这里写自定义目录标题 如果使用Vue要做根据已有的图形填入到指定的单元格中&#xff0c;你会怎么做&#xff1f;先看效果图上代码 如果使用Vue要做根据已有的图形填入到指定的单元格中&#xff0c;你会怎么做&#xff1f; 先看效果图 上代码 <template><div class&q…

代理IP怎么使用

使用代理IP的步骤如下&#xff1a; 获取代理IP地址&#xff1a;从代理IP服务商处获取代理IP地址。打开浏览器&#xff1a;打开浏览器&#xff0c;并进入代理设置页面。设置代理IP&#xff1a;在代理设置页面中&#xff0c;将代理IP地址和端口号填写到对应的文本框中&#xff0…

kubernetes集群编排——prometheus监控

部署prometheus 创建项目仓库并上传镜像 编写配置文件 [rootk8s2 values]# vim prometheus-values.yaml alertmanager:alertmanagerSpec:image:repository: prometheus/alertmanagertag: v0.24.0 grafana:enabled: trueimage:repository: grafana/grafanatag: 9.0.6service:typ…

Oneid方案

一、前文 用户画像的前提是标识出用户&#xff0c;存在以下场景&#xff1a;不同业务系统对同一个人的标识&#xff0c;匿名用户行为的行为归因&#xff1b;本文提供多种解决方案&#xff0c;提供大家思考。 二、方案矩阵 三、其他 相关连接&#xff1a; 如何通过图算法能力获…

Linux权限管理

目录 前言 1. Linux权限的概念 2. Linux权限管理 2.1 修改权限 拓展 2.2 修改拥有者 2.3 修改所属组 3. 文件类型 3.1 file指令 4. 权限掩码 umask指令 5.目录权限 6. 粘滞位 总结 前言 Linux作为一款开源操作系统&#xff0c;其权限管理机制是非常重要的一部分。熟练掌…

新一轮SocialFi浪潮来袭,Atem Network 再次打响注意力争夺战

火爆如潮的 Atem Network 再次从 CyberConnect 以及 Friend.tech 手中接过 SocialFi 赛道的热度大棒&#xff0c;同时这也表明&#xff0c;协议层仍将是 Web3 社交领域的主要叙事。 前不久&#xff0c;Web3社交协议Atem Network 在白皮书中披露了ATEM的代币经济模型&#xff0c…

基础课4——客服中心管理者面临的挑战

客服管理者在当今的数字化时代也面临着许多挑战。以下是一些主要的挑战&#xff1a; 同行业竞争加剧&#xff1a;客服行业面临着来自同行业的竞争压力。为了获得竞争优势&#xff0c;企业需要不断提高自身的产品和服务质量&#xff0c;同时还需要不断降低成本、提高效率。然而…