SpringSecurity之SecurityContextHolder使用解讀
之前在使用SpringSecurity的過程中并沒有把很多東西理解透徹,找了很多學習資料也都只是是很淺顯了告訴你怎么使用這個東西。現(xiàn)在只能自己回過頭來研究研究。。。。終究是一個人扛下了所有/(ㄒoㄒ)/~~。
GO,現(xiàn)在越看spring源碼越覺得里面注釋是真的詳細,雖然英語很菜耐不住有翻譯哈哈哈。
介紹
看了幾篇大佬的技術博客里面都介紹到,SecurityContextHolder是保存全上下文對象(SecurityContext)的地方。對于這種說法我感覺是完全不正確的,如果后面有新的認知我會回來進行修改。
我認為SecurityContextHolder只是為SecurityContext提供一種存儲策略,只是主導了他的存儲方式及地址。
使用中我們表面上看起來SecurityContext的存儲都是通過SecurityContextHolder在控制人們就習以為常的說成了Security Context存儲在了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提供了一個SecurityContextHolderStrategy存儲策略進行上下文的存儲,進入到Security ContextHolderStrategy接口,由下圖我們可以清晰的看到其總共有三個實現(xiàn)類。
分別對應三種存儲策略,這里我不知道為什么99的文章都只說了threadlocal和global兩種。
要不是看了下實現(xiàn)類我都信了。
就跟字面意思一樣三種策略分別對應threadlocal,global,InheritableThreadLocal三種方式。
繼續(xù)回到源碼我們詳細看這三種方式。
private static void initialize() { if (!StringUtils.hasText(strategyName)) { // 如果沒有設置自定義的策略,就采用MODE_THREADLOCAL模式 strategyName = MODE_THREADLOCAL; } // ThreadLocal策略 if (strategyName.equals(MODE_THREADLOCAL)) { strategy = new ThreadLocalSecurityContextHolderStrategy(); } // 采用InheritableThreadLocal,它是ThreadLocal的一個子類 else if (strategyName.equals(MODE_INHERITABLETHREADLOCAL)) { strategy = new InheritableThreadLocalSecurityContextHolderStrategy(); } // 全局策略,實現(xiàn)方式就是static SecurityContext contextHolder else if (strategyName.equals(MODE_GLOBAL)) { strategy = new GlobalSecurityContextHolderStrategy(); } else { // Try to load a custom strategy 自定義的策略,通過返回創(chuàng)建出 try { Class<?> clazz = Class.forName(strategyName); Constructor<?> customStrategy = clazz.getConstructor(); strategy = (SecurityContextHolderStrategy) customStrategy.newInstance(); } catch (Exception ex) { ReflectionUtils.handleReflectionException(ex); } } initializeCount++; }
由源碼我們可以得出SecurityContextHolder 默認使用的是THREADLOCAL模式,光從名字看感覺就是存儲在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; 。。。。。。 }
到這里那些說SecurityContext存儲在SecurityContextHolder的大佬們我認為應該是不嚴謹?shù)摹?/p>
默認是將SecurityContext存儲在threadlocal中,可能是spring考慮到目前大多數(shù)為BS應用,一個應用同時可能有多個使用者,每個使用者又對應不同的安全上下,Security Context Holder為了保存這些安全上下文。
缺省情況下,使用了ThreadLocal機制來保存每個使用者的安全上下文。
因為缺省情況下根據(jù)Servlet規(guī)范,一個Servlet request的處理不管經(jīng)歷了多少個Filter,自始至終都由同一個線程來完成。這樣就很好的保證了其安全性。
但是當我們開發(fā)的是一個CS本地應用的時候,這種模式就不太適用了。
spring早早的就考慮到了這種情況,這個時候我們就可以設置為Global模式僅使用一個變量來存儲SecurityContext。比如還有其他的一些應用會有自己的線程創(chuàng)建,并且希望這些新建線程也能使用創(chuàng)建者的安全上下文。
這種效果,我們就可以通過將SecurityContextHolder配置成MODE_INHERITABLETHREADLOCAL策略達到。
SecurityContext又到底是個什么東西呢?稍后新開一遍學習記錄。
總結
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關文章
如何從官網(wǎng)下載Hibernate jar包的方法示例
這篇文章主要介紹了如何從官網(wǎng)下載Hibernate jar包的方法示例,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2019-04-04Java編程實現(xiàn)基于TCP協(xié)議的Socket聊天室示例
這篇文章主要介紹了Java編程實現(xiàn)基于TCP協(xié)議的Socket聊天室,結合實例形式詳細分析了java基于TCP協(xié)議的Socket聊天室客戶端與服務器端相關實現(xiàn)與使用技巧,需要的朋友可以參考下2018-01-01SpringBoot 整合 Netty 多端口監(jiān)聽的操作方法
Netty提供異步的、基于事件驅動的網(wǎng)絡應用程序框架,用以快速開發(fā)高性能、高可靠性的網(wǎng)絡 IO 程序,是目前最流行的 NIO 框架,這篇文章主要介紹了SpringBoot 整和 Netty 并監(jiān)聽多端口,需要的朋友可以參考下2023-10-10