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.mzph.cn/news/143503.shtml

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

相关文章

【函数】各种函数收集,持续更新~

文章目录 各种函数收集&#xff0c;持续更新~1. 数组对象去重2. 时间转换函数3. 深拷贝4. js精度问题解决&#xff08;加减乘除&#xff09;5. toFiexed函数重写6.手机号校验与隐藏手机号中间4位7. 身份证号校验与加密校验位检测15位转18位身份证8. 千位符格式化&#xff08;12…

anaconda中安装pytorch和TensorFlow环境并在不同环境中安装kernel

❤️觉得内容不错的话&#xff0c;欢迎点赞收藏加关注&#x1f60a;&#x1f60a;&#x1f60a;&#xff0c;后续会继续输入更多优质内容❤️ &#x1f449;有问题欢迎大家加关注私戳或者评论&#xff08;包括但不限于NLP算法相关&#xff0c;linux学习相关&#xff0c;读研读博…

单链表的实现

CSDN主页&#xff1a;醋溜马桶圈_C语言进阶,初始C语言,数据结构-CSDN博客 Gitee主页&#xff1a;mnxcc (mnxcc) - Gitee.com 专栏&#xff1a;数据结构_醋溜马桶圈的博客-CSDN博客 目录 1.认识单链表 2.创建单链表 3.单链表的操作 3.1打印单链表 3.2开辟新空间 3.3尾插 3.4头插…

深入探讨Linux中的文本文件查看命令

目录 前言1 cat命令2 less命令3 more命令4 head命令5 tail命令6 总结 前言 在Linux系统中&#xff0c;文本文件是日常工作中不可或缺的一部分&#xff0c;无论是配置文件、日志文件还是代码文件&#xff0c;都需要用到文本文件查看命令。在本文中&#xff0c;我们将深入研究一…

AI创作系统ChatGPT源码+AI绘画系统+支持OpenAI DALL-E3文生图,可直接对话文生图

一、AI创作系统 SparkAi创作系统是基于OpenAI很火的ChatGPT进行开发的Ai智能问答系统和Midjourney绘画系统&#xff0c;支持OpenAI-GPT全模型国内AI全模型。本期针对源码系统整体测试下来非常完美&#xff0c;可以说SparkAi是目前国内一款的ChatGPT对接OpenAI软件系统。新增支…

桶装水订水小程序app,线上预约订水更便捷

桶装水订水小程序app&#xff0c;线上预约订水更便捷。设置好地址&#xff0c;一键订水&#xff0c;工作人员送水到家。还能配送新鲜果蔬&#xff0c;绿色健康有保证。送水软件手机版&#xff0c;提供各种品牌桶装水&#xff0c;在线发起订水服务&#xff0c;由服务人员送水到家…

QT windows与linux之间sokcet通信中文乱码问题解决方法

QT windows与linux之间sokcet通信中文乱码问题解决方法 linux发送与接收都转码utf-8: tcpClient ->write( send_msg.toUtf8());//解决乱码&#xff0c;发送转码 接收&#xff1a; QByteArray buffer tcpClient->readAll(); if(!buffer.isEmpty()) { // ui->plain…

ipad可能会在iOS 16中失去智能家居中心功能

在iOS 16测试版代码中发现的文本表明苹果将放弃对iPad家庭中心的支持 家庭app迎来重大改版&#xff0c;未来更将对智能家居互联互通标准Matter提供支持。 即使某一款智能家居设备再优秀&#xff0c;只要它没有接入HomeKit&#xff0c;那么就不能在苹果的家庭app中直接管理控制。…

双十一买高画质投影仪,当贝F6还是极米H6?

如果你想购买一台4K画质的投影仪,那么在各大平台搜索“双十一最值得买的4K投影仪”时,一定会注意到当贝F6和极米H6这两个型号投影仪。个人认为当贝F6和极米H6都分别是当贝和极米两个品牌非常具有性价比的4K投影仪。那么到底哪一台更适合你。 首先放一张参数对比图,方便参数控研…

登上CMMLU性能评测榜单第一 四大维度解码夸克自研大模型

11月14日&#xff0c;拥有千亿参数的夸克自研大模型正式发布&#xff0c;立刻占据CMMLU榜单第一名。夸克大模型将应用于通用搜索、医疗健康、教育学习、职场办公等多个场景。性能方面&#xff0c;其整体水平已经超过GPT-3.5&#xff0c;其中在写作、考试等部分场景中可以超过GP…

OpenCV实现手势虚拟拖拽

前言&#xff1a; Hello大家好&#xff0c;我是Dream。 今天来学习一下如何使用OpenCV实现手势虚拟拖拽&#xff0c;欢迎大家一起前来探讨学习~ 一、主要步骤及库的功能介绍 1.主要步骤 要实现本次实验&#xff0c;主要步骤如下&#xff1a; 导入OpenCV库。通过OpenCV读取摄…

解决element的el-input无法输入问题

el-inputmaxlength"5"placeholder"请输入年龄"v-model"addValue.age"input"changeMessage"onkeyup"this.valuethis.value.replace(/^0/g, ).replace(/[^0-9]/g, )" > </el-input>changeMessage(){//强制刷新或者使…

做决策、定战略、带团队:顶级高手常用的16个思维模型

01 做决策 1.沃伦巴菲特的双目标清单系统&#xff08;Two-List System&#xff09; 弗林特当了巴菲特的私人飞行员十年之久&#xff0c;还曾为美国四任总统开过飞机&#xff0c;但他在事业上依然有更多追求。有一次&#xff0c;他和巴菲特在探讨他的职业生涯目标时&#xff0…

Node.js中的Buffer和Stream

Node.js中的Buffer和Stream 计算机只能理解二进制数据&#xff0c;即0和1形式的数据。这些数据的顺序移动称为流。以称为块&#xff08;chunk&#xff09;的破碎部分流式传输数据&#xff1b;计算机一收到数据块就开始处理数据&#xff0c;而不用等待整个数据。 我们这篇文章…

python类接口和抽象超类

1 python类接口和抽象超类 1.1 导入模块执行class语句 python导入模块时会执行class语句及主体内的顶层语句。 示例 # test.py class MyClass:print(MyClass) # cmd执行下面语句 E:\documents\F盘>python >>> import test MyClass1.2 pytho类接口技术 python类…

容器化nacos部署并实现服务发现(gradle)

1.如何容器化部署mysql 2. 如何容器化部署nacos 为不暴露我的服务器地址&#xff0c;本文全部使用localhost来代替服务器地址&#xff0c;所有的localhost都应该调整为你自己的服务器地址。 为不暴露我的服务器地址&#xff0c;本文全部使用localhost来代替服务器地址&#x…

台式电脑的IP地址在哪里?解密台式电脑网络连接的秘密!

​ 如今智能手机和便携式笔记本电脑盛行的时代&#xff0c;台式电脑似乎逐渐被人们所忽视。然而&#xff0c;对于需要高性能和大容量存储的用户来说&#xff0c;台式电脑依然是最好的选择。而作为一款能够连接网络的设备&#xff0c;台式电脑也有自己独特的IP地址。下面虎观代理…

我自己理解的JAVA反射

1.定义 JAVA反射机制是在运行状态中&#xff0c;对于任意一个类&#xff0c;都能够知道这个类的所有属性和方法&#xff1b;对于任意一个对象&#xff0c;都能够调用它的任意一个方法和属性&#xff1b;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制…

Python之函数进阶-匿名函数

Python之函数进阶-匿名函数 匿名函数 匿名:隐藏名字&#xff0c;即没有名称匿名函数:没有名字的函数。Lambda表达式 Python中&#xff0c;使用Lambda表达式构建匿名函数。 使用lambda关键字定义匿名函数&#xff0c;格式为 lambda [参数列表]: 表达式参数列表不需要小括号。无…

Libra R-CNN: Towards Balanced Learning for Object Detection(2019.4)

文章目录 AbstractIntroduction引入问题1&#xff09; Sample level imbalance2) Feature level imbalance3) Objective level imbalance进行解决贡献 Related Work&#xff08;他人的work&#xff0c;捎带与我们的对比&#xff09;Model architectures for object detection&a…