Spring如何解決單例bean線程不安全的問題
首先我們應該知道線程安全問題一般發生在成員變量上,這是為什么啦?
因為成員變量是存放在堆內存中,而堆內存又是線程共享的,這就造成了線程安全問題
因為Spring中的Bean默認是單例的,所以在定義成員變量時也有可能會發生線程安全問題。下面我們就來研究下如何解決Spring中單例Bean的線程安全問題
@RestController//@Scope('prototype')public class BeanController { private int content=0; //基本類型 線程不安全 private String test=null;//引用類型 線程不安全 @RequestMapping('testBean') public Object getSercurity(){ System.out.println(content); System.out.println(test); content=20; test='單例模式是不安全的'; return test; }
問題來了,我們該如何測試線程不安全問題啦?我們需要在程序中用debug模式去啟動,打斷點。不需要執行完程序,然后再次調用該接口。或者多次調用該接口,便會出現以下控制臺所示的結果。
下面我們就來討論下解決這個線程不安全的問題的辦法
解決方式一:
在對應的類名上加上該注解@Scope('prototype'),表示每次調用該接口都會生成一個新的Bean。下圖示例
解決方案二 ThreadLocal解決問題
@RestController//@Scope('prototype')public class BeanController { private static ThreadLocal<Integer> content = new ThreadLocal<Integer>() { @Override protected Integer initialValue() { return (int)(Math.random()*10+100); } }; private static ThreadLocal<String> test = new ThreadLocal<String>() { @Override protected String initialValue() { return '單例模式是不安全的'+(int)(Math.random()*10+100); } }; @RequestMapping('testBean') public Object getSercurity(){ System.out.println(content.get()); System.out.println(test.get()); System.out.println(); return test.get(); }}
第三種解決方案:
盡量不要使用成員變量
第四種解決方案:
前提:
該程序是web應用,可以使用Spring Bean的作用域中的request,就是說在類前面加上@Scope('request'),表明每次請求都會生成一個新的Bean對象。
作用于@Scope('prototype')類似。
補充知識:SpringMVC是單例的,高并發情況下,如何保證性能的?
首先在大家的思考中,肯定有影響的,你想想,單例顧名思義:一個個排隊過... 高訪問量的時候,你能想象服務器的壓力了... 而且用戶體驗也不怎么好,等待太久~
實質上這種理解是錯誤的,Java里有個API叫做ThreadLocal,spring單例模式下用它來切換不同線程之間的參數。用ThreadLocal是為了保證線程安全,實際上ThreadLoacal的key就是當前線程的Thread實例。單例模式下,spring把每個線程可能存在線程安全問題的參數值放進了ThreadLocal。這樣雖然是一個實例在操作,但是不同線程下的數據互相之間都是隔離的,因為運行時創建和銷毀的bean大大減少了,所以大多數場景下這種方式對內存資源的消耗較少,而且并發越高優勢越明顯。
總的來說就是,單利模式因為大大節省了實例的創建和銷毀,有利于提高性能,而ThreadLocal用來保證線程安全性。
另外補充說一句,單例模式是spring推薦的配置,它在高并發下能極大的節省資源,提高服務抗壓能力。spring IOC的bean管理器是“絕對的線程安全”。
以上這篇Spring如何解決單例bean線程不安全的問題就是小編分享給大家的全部內容了,希望能給大家一個參考,也希望大家多多支持好吧啦網。
相關文章: