SpringSecurity之SecurityContextHolder使用解讀
之前在使用SpringSecurity的過(guò)程中并沒(méi)有把很多東西理解透徹,找了很多學(xué)習(xí)資料也都只是是很淺顯了告訴你怎么使用這個(gè)東西?,F(xiàn)在只能自己回過(guò)頭來(lái)研究研究。。。。終究是一個(gè)人扛下了所有/(ㄒoㄒ)/~~。
GO,現(xiàn)在越看spring源碼越覺(jué)得里面注釋是真的詳細(xì),雖然英語(yǔ)很菜耐不住有翻譯哈哈哈。
介紹
看了幾篇大佬的技術(shù)博客里面都介紹到,SecurityContextHolder是保存全上下文對(duì)象(SecurityContext)的地方。對(duì)于這種說(shuō)法我感覺(jué)是完全不正確的,如果后面有新的認(rèn)知我會(huì)回來(lái)進(jìn)行修改。
我認(rèn)為SecurityContextHolder只是為SecurityContext提供一種存儲(chǔ)策略,只是主導(dǎo)了他的存儲(chǔ)方式及地址。
使用中我們表面上看起來(lái)SecurityContext的存儲(chǔ)都是通過(guò)SecurityContextHolder在控制人們就習(xí)以為常的說(shuō)成了Security Context存儲(chǔ)在了SecurityContextHolder中,至于真相我們看源碼慢慢分析。
public class SecurityContextHolder { // ~ Static fields/initializers // ===================================================================================== public static final String MODE_THREADLOCAL = "MODE_THREADLOCAL"; public static final String MODE_INHERITABLETHREADLOCAL = "MODE_INHERITABLETHREADLOCAL"; public static final String MODE_GLOBAL = "MODE_GLOBAL"; public static final String SYSTEM_PROPERTY = "spring.security.strategy"; private static String strategyName = System.getProperty(SYSTEM_PROPERTY); private static SecurityContextHolderStrategy strategy; private static int initializeCount = 0;
首先從源碼可以看到SecurityContextHolder提供了一個(gè)SecurityContextHolderStrategy存儲(chǔ)策略進(jìn)行上下文的存儲(chǔ),進(jìn)入到Security ContextHolderStrategy接口,由下圖我們可以清晰的看到其總共有三個(gè)實(shí)現(xiàn)類(lèi)。
分別對(duì)應(yīng)三種存儲(chǔ)策略,這里我不知道為什么99的文章都只說(shuō)了threadlocal和global兩種。
要不是看了下實(shí)現(xiàn)類(lèi)我都信了。
就跟字面意思一樣三種策略分別對(duì)應(yīng)threadlocal,global,InheritableThreadLocal三種方式。
繼續(xù)回到源碼我們?cè)敿?xì)看這三種方式。
private static void initialize() { if (!StringUtils.hasText(strategyName)) { // 如果沒(méi)有設(shè)置自定義的策略,就采用MODE_THREADLOCAL模式 strategyName = MODE_THREADLOCAL; } // ThreadLocal策略 if (strategyName.equals(MODE_THREADLOCAL)) { strategy = new ThreadLocalSecurityContextHolderStrategy(); } // 采用InheritableThreadLocal,它是ThreadLocal的一個(gè)子類(lèi) else if (strategyName.equals(MODE_INHERITABLETHREADLOCAL)) { strategy = new InheritableThreadLocalSecurityContextHolderStrategy(); } // 全局策略,實(shí)現(xiàn)方式就是static SecurityContext contextHolder else if (strategyName.equals(MODE_GLOBAL)) { strategy = new GlobalSecurityContextHolderStrategy(); } else { // Try to load a custom strategy 自定義的策略,通過(guò)返回創(chuàng)建出 try { Class<?> clazz = Class.forName(strategyName); Constructor<?> customStrategy = clazz.getConstructor(); strategy = (SecurityContextHolderStrategy) customStrategy.newInstance(); } catch (Exception ex) { ReflectionUtils.handleReflectionException(ex); } } initializeCount++; }
由源碼我們可以得出SecurityContextHolder 默認(rèn)使用的是THREADLOCAL模式,光從名字看感覺(jué)就是存儲(chǔ)在threadlocal中的策略到底是不是我們?nèi)ピ创a中分別一探究竟:
ThreadLocalSecurityContextHolderStrategy
final class ThreadLocalSecurityContextHolderStrategy implements SecurityContextHolderStrategy { // ~ Static fields/initializers // ===================================================================================== private static final ThreadLocal<SecurityContext> contextHolder = new ThreadLocal<>(); 。。。。。。 }
InheritableThreadLocalSecurityContextHolderStrategy
final class InheritableThreadLocalSecurityContextHolderStrategy implements SecurityContextHolderStrategy { // ~ Static fields/initializers // ===================================================================================== private static final ThreadLocal<SecurityContext> contextHolder = new InheritableThreadLocal<>(); 。。。。。。 }
GlobalSecurityContextHolderStrategy
final class GlobalSecurityContextHolderStrategy implements SecurityContextHolderStrategy { // ~ Static fields/initializers // ===================================================================================== private static SecurityContext contextHolder; 。。。。。。 }
到這里那些說(shuō)SecurityContext存儲(chǔ)在SecurityContextHolder的大佬們我認(rèn)為應(yīng)該是不嚴(yán)謹(jǐn)?shù)摹?/p>
默認(rèn)是將SecurityContext存儲(chǔ)在threadlocal中,可能是spring考慮到目前大多數(shù)為BS應(yīng)用,一個(gè)應(yīng)用同時(shí)可能有多個(gè)使用者,每個(gè)使用者又對(duì)應(yīng)不同的安全上下,Security Context Holder為了保存這些安全上下文。
缺省情況下,使用了ThreadLocal機(jī)制來(lái)保存每個(gè)使用者的安全上下文。
因?yàn)槿笔∏闆r下根據(jù)Servlet規(guī)范,一個(gè)Servlet request的處理不管經(jīng)歷了多少個(gè)Filter,自始至終都由同一個(gè)線(xiàn)程來(lái)完成。這樣就很好的保證了其安全性。
但是當(dāng)我們開(kāi)發(fā)的是一個(gè)CS本地應(yīng)用的時(shí)候,這種模式就不太適用了。
spring早早的就考慮到了這種情況,這個(gè)時(shí)候我們就可以設(shè)置為Global模式僅使用一個(gè)變量來(lái)存儲(chǔ)SecurityContext。比如還有其他的一些應(yīng)用會(huì)有自己的線(xiàn)程創(chuàng)建,并且希望這些新建線(xiàn)程也能使用創(chuàng)建者的安全上下文。
這種效果,我們就可以通過(guò)將SecurityContextHolder配置成MODE_INHERITABLETHREADLOCAL策略達(dá)到。
SecurityContext又到底是個(gè)什么東西呢?稍后新開(kāi)一遍學(xué)習(xí)記錄。
總結(jié)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
Springboot實(shí)現(xiàn)VNC的反向代理功能
這篇文章主要介紹了Springboot實(shí)現(xiàn)VNC的反向代理,搭建過(guò)程也很簡(jiǎn)單,通過(guò)注冊(cè)bean攔截指定URL路徑進(jìn)行自定義操作,具體實(shí)例代碼跟隨小編一起看看需要的朋友可以參考下2021-09-09如何從官網(wǎng)下載Hibernate jar包的方法示例
這篇文章主要介紹了如何從官網(wǎng)下載Hibernate jar包的方法示例,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2019-04-04IDEA調(diào)試技巧條件斷點(diǎn)實(shí)現(xiàn)步驟詳解
這篇文章主要介紹了IDEA調(diào)試技巧條件斷點(diǎn)實(shí)現(xiàn)步驟詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-09-09Java編程實(shí)現(xiàn)基于TCP協(xié)議的Socket聊天室示例
這篇文章主要介紹了Java編程實(shí)現(xiàn)基于TCP協(xié)議的Socket聊天室,結(jié)合實(shí)例形式詳細(xì)分析了java基于TCP協(xié)議的Socket聊天室客戶(hù)端與服務(wù)器端相關(guān)實(shí)現(xiàn)與使用技巧,需要的朋友可以參考下2018-01-01Java線(xiàn)程Timer定時(shí)器用法詳細(xì)總結(jié)
在本篇文章里小編給大家整理的是關(guān)于Java線(xiàn)程Timer定時(shí)器用法詳細(xì)總結(jié)內(nèi)容,需要的朋友們學(xué)習(xí)下吧。2020-02-02Windows下Java調(diào)用可執(zhí)行文件代碼實(shí)例
這篇文章主要介紹了Windows下Java調(diào)用可執(zhí)行文件代碼實(shí)例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-12-12SpringBoot 整合 Netty 多端口監(jiān)聽(tīng)的操作方法
Netty提供異步的、基于事件驅(qū)動(dòng)的網(wǎng)絡(luò)應(yīng)用程序框架,用以快速開(kāi)發(fā)高性能、高可靠性的網(wǎng)絡(luò) IO 程序,是目前最流行的 NIO 框架,這篇文章主要介紹了SpringBoot 整和 Netty 并監(jiān)聽(tīng)多端口,需要的朋友可以參考下2023-10-10SpringBoot統(tǒng)計(jì)接口調(diào)用耗時(shí)的三種方式
在實(shí)際開(kāi)發(fā)中,了解項(xiàng)目中接口的響應(yīng)時(shí)間是必不可少的事情,SpringBoot 項(xiàng)目支持監(jiān)聽(tīng)接口的功能也不止一個(gè),接下來(lái)我們分別以 AOP、ApplicationListener、Tomcat 三個(gè)方面去實(shí)現(xiàn)三種不同的監(jiān)聽(tīng)接口響應(yīng)時(shí)間的操作,需要的朋友可以參考下2024-06-06Spring和SpringMVC父子容器關(guān)系初窺(小結(jié))
這篇文章主要介紹了Spring和SpringMVC父子容器關(guān)系初窺(小結(jié)),小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-01-01