Java 單例模式線程安全問題
Java 單例模式線程安全問題
SpringIOC容器默認提供bean的訪問作用域是單例模式。即在整個application生命周期中,只有一個instance。因此在多線程并發(fā)下,會有線程安全風險。我們在MVC框架下的servlet就是線程安全的。由于該servlet是在客戶端,多并發(fā)相對少,但是對于web service端,需要考慮到。
ThreadLocal類:為每一個線程提供了一個獨立的變量(實例)副本,從各將各個不同的實例訪問isolation。
在同步鎖機制中,后來者線程等待先行線程完成后,才能訪問該成員變量。而ThreadLocal實現(xiàn)實例的復制而隔離對象訪問數(shù)據(jù)沖突。同時也能夠少量解決scope為prototype訪問模式下大量實例生命周期管理的消耗和負擔。是“以時間換空間”和“以空間換時間”的兩種實現(xiàn)。前者僅提供唯一變量,讓不同的線程排隊訪問,而后者為每一個線程都提供了一份拷貝,因此可以同時訪問而互不影響,同時該拷貝存儲于內(nèi)存中,下次再次訪問時候不再重新生成實例,減少服務器資源消耗。
我們知道在一般情況下,只有無狀態(tài)的Bean才可以在多線程環(huán)境下共享,在Spring中,絕大部分Bean都可以聲明為singleton作用域。就是因為Spring對一些Bean(如RequestContextHolder、TransactionSynchronizationManager、LocaleContextHolder等)中非線程安全狀態(tài)采用ThreadLocal進行處理,讓它們也成為線程安全的狀態(tài),因為有狀態(tài)的Bean就可以在多線程中共享了。
線程安全問題:由全局變量及靜態(tài)變量引起,若每個線程中對全局變量、靜態(tài)變量只有讀操作,而無寫操作,一般來說,這個全局變量是線程安全的;若有多個線程同時執(zhí)行寫操作,一般都需要考慮線程同步,否則就可能影響線程安全。
1) 常量始終是線程安全的 (值恒定)
2)每次調(diào)用方法前都新建一個實例是線程安全的。(不同實例相互隔離)
3)局部變量是線程安全的(隔離)
因為每執(zhí)行一個方法,都會在獨立的空間創(chuàng)建局部變量,它不是共享的資源。局部變量包括方法的參數(shù)變量和方法內(nèi)變量。
有狀態(tài):有數(shù)據(jù)存儲、更改功能。有狀態(tài)對象(Stateful Bean),有實例變量的對象 ,可以保存數(shù)據(jù),是非線程安全的。
無狀態(tài):就是一次操作,不能變更數(shù)據(jù)。無狀態(tài)對象(Stateless Bean),沒有實例變量的對象,不能保存數(shù)據(jù),是不變類,是線程安全的。在spring中單例模式是共享實例,以達到提高性能。有狀態(tài)的Bean,多線程環(huán)境下不安全,那么適合用Prototype原型模式。Prototype: 每次對bean的請求都會創(chuàng)建一個新的bean實例。
如有疑問請留言或者到本站社區(qū)交流討論,感謝閱讀,希望能幫助到大家,謝謝大家對本站的支持!
相關文章
springboot業(yè)務功能實戰(zhàn)之告別輪詢websocket的集成使用
WebSocket使得客戶端和服務器之間的數(shù)據(jù)交換變得更加簡單,允許服務端主動向客戶端推送數(shù)據(jù),下面這篇文章主要給大家介紹了關于springboot業(yè)務功能實戰(zhàn)之告別輪詢websocket的集成使用,需要的朋友可以參考下2022-10-10idea中maven本地倉庫jar包打包失敗和無法引用的問題解決
本文主要介紹了idea中maven本地倉庫jar包打包失敗和無法引用的問題解決,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2023-06-06Spring使用@Autowired為抽象父類注入依賴代碼實例
這篇文章主要介紹了Spring使用@Autowired為抽象父類注入依賴代碼實例,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下2019-11-11