Springboot注入成員變量HttpServletRequest的原理分析
前言
最近做項目,springboot項目,本來我們在controller的requestmapping取參數(shù)值或者返回寫時,使用方法參數(shù),但是發(fā)現(xiàn)老項目直接注入了成員變量,Spring本身是單例的,如果是成員變量注入,那么也是單例的,怎么實現(xiàn)不同的請求讀取不同的參數(shù)呢,如果實現(xiàn)線程安全呢,筆者立馬想到了ThreadLocal,但是如果要說就是這個原理,那么必須源碼證明。
準備demo
簡單寫一個demo
@RestController public class DemoController { @Autowired private HttpServletRequest request; @GetMapping("/hello") public String demo(String param) { request.getParameterMap().forEach((k,v)-> System.out.println(k + " : " + Arrays.toString(v))); return param + ":hello"; } }
只寫了 HttpServletRequest,實際上HttpServletResponse亦是如此。
分析demo,注入的HttpServletRequest是接口類型,那么在Boot啟動中就會動態(tài)代理實現(xiàn),由于是接口,可以推測是JDK動態(tài)代理,debug果然如此。
源碼分析
根據(jù)debug,JDK動態(tài)代理注入了接口的實現(xiàn)類,關(guān)鍵在于InvocationHandler
Spring使用如下:
org.springframework.beans.factory.support.AutowireUtils.ObjectFactoryDelegatingInvocationHandler
里面有關(guān)鍵代碼
return method.invoke(this.objectFactory.getObject(), args);
this.objectFactory.getObject()這句決定線程安全
看看Spring Bean下JDK是怎么動態(tài)代理注入的
可以看到JDK動態(tài)代理在Spring注入的時候,把這個factory注入了InvocationHandler
其中的handler的invoke方法,這里實際上還要其他類的讀取埋點。
這里的invoke僅僅是反射,關(guān)鍵還是 HttpServletRequest的對象來源,跟蹤讀取邏輯
org.springframework.web.context.request.RequestContextHolder
到此就明確了,注入的成員變量,動態(tài)代理后使用Threadlocal處理,所以線程安全,因為每次請求都是線程請求,那么原始的HttpServletRequest對象怎么塞進去的呢,就要看filter的了
org.springframework.web.filter.RequestContextFilter
在doFilter時,執(zhí)行
同理在org.springframework.web.servlet.FrameworkServlet也會再次讀取和寫入
這里是為了,如果filter被去除的時候可以有值,再次保底,并且在結(jié)束時rest
只不過這個rest有點奇怪
對象并沒有清除,還在,說明即使 FrameworkServlet后還可以獲取,因為有filter可能會在這個后面執(zhí)行,如果干掉,很可能就不能讀取了。
會在RequestContextFilter后面remove;如果我們?nèi)サ暨@個filter,那么需要自定義一個filter實現(xiàn)remove防止內(nèi)存泄漏。
清除當前線程及子線程ThreadLocal
public static void resetRequestAttributes() { requestAttributesHolder.remove(); inheritableRequestAttributesHolder.remove(); }
至此 HttpServletRequest的成員變量注入邏輯,即ThreadLocal變量,所以請求可以正常訪問
總結(jié)
實際上這種用法很多項目都用了,只不過我們寫代碼下意思的通過方法參數(shù)來規(guī)避線程安全性,這種想法是有益的,可以從源頭規(guī)避風險,不過實際上Spring也幫我們考慮了這個問題,相當于使用RequestContextFilter做了保底措施。
源碼分析實際上是知所以然,尤其是我們自己寫公共SDK時,可以把這種設計放在代碼中,實現(xiàn)優(yōu)雅和保底邏輯。
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
- SpringBoot集成WebServlet出現(xiàn)自定義servlet請求失敗的問題解決方案
- SpringBoot里使用Servlet進行請求的實現(xiàn)示例
- springboot掃描自定義的servlet和filter代碼詳解
- SpringBoot3.1.2 引入Swagger報錯Type javax.servlet.http.HttpServletRequest not present解決辦法
- 解決IDEA啟動springboot項目報錯java.lang.ClassNotFoundException:?javax.servlet.ServletContext
- SpringBoot獲取HttpServletRequest的3種方式總結(jié)
- Springboot如何添加server.servlet.context-path相關(guān)使用
- SpringBoot項目找不到javax.servlet.Filter的問題及解決
- SpringBoot如何切換成其它的嵌入式Servlet容器(Jetty和Undertow)
相關(guān)文章
Spring Boot2深入分析解決java.lang.ArrayStoreException異常
這篇文章介紹了Spring Boot2深入分析解決java.lang.ArrayStoreException異常的方法,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2021-12-12java Socket實現(xiàn)簡單模擬HTTP服務器
這篇文章主要介紹了java Socket實現(xiàn)簡單模擬HTTP服務器,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-05-05利用Spring Social輕松搞定微信授權(quán)登錄的方法示例
這篇文章主要介紹了利用Spring Social輕松搞定微信授權(quán)登錄的方法示例,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-12-12MyBatis-Plus 之selectMaps、selectObjs、selectCount、selectO
本文主要介紹了MyBatis-Plus 之selectMaps、selectObjs、selectCount、selectOne的使用,文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2022-03-03Springmvc如何實現(xiàn)向前臺傳遞數(shù)據(jù)
這篇文章主要介紹了Springmvc如何實現(xiàn)向前臺傳遞數(shù)據(jù),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下2020-07-07Springboot Cache @CacheEvict 無法模糊刪除的解決方案
這篇文章主要介紹了Springboot Cache @CacheEvict 無法模糊刪除的解決方案,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-12-12