天天看點

Spring 中的 bean 為什麼預設單例?

熟悉Spring開發的朋友都知道Spring提供了5種scope分别是singleton、prototype、request、session、global session。

如下圖是官方文檔上的截圖,感興趣的朋友可以進去看看這五種分别有什麼不同。

今天要介紹的是這五種中的前兩種,也是Spring最初提供的bean scope singleton 和 prototype。

Spring官方文檔介紹如下圖:

Spring 中的 bean 為什麼預設單例?

更多内容可以看官方文檔介紹,非常詳細:

https://docs.spring.io/spring/docs/3.0.0.M3/reference/html/ch04s04.html

單例bean與原型bean的差別

如果一個bean被聲明為單例的時候,在處理多次請求的時候在Spring容器裡隻執行個體化出一個bean,後續的請求都公用這個對象,這個對象會儲存在一個map裡面。

當有請求來的時候會先從緩存(map)裡檢視有沒有,有的話直接使用這個對象,沒有的話才執行個體化一個新的對象,是以這是個單例的。

但是對于原型(prototype)bean來說當每次請求來的時候直接執行個體化新的bean,沒有緩存以及從緩存查的過程。

###

1.畫圖分析

Spring 中的 bean 為什麼預設單例?
Spring 中的 bean 為什麼預設單例?

2.源碼分析

生成bean時先判斷單例的還是原型的

Spring 中的 bean 為什麼預設單例?

如果是單例的則先嘗試從緩存裡擷取,沒有在新建立

Spring 中的 bean 為什麼預設單例?

結論:

單例的bean隻有第一次建立新的bean 後面都會複用該bean,是以不會頻繁建立對象。

原型的bean每次都會新建立

單例bean的優勢

由于不會每次都新建立新對象是以有一下幾個性能上的優勢:

1.減少了新生成執行個體的消耗

新生成執行個體消耗包括兩方面,第一,Spring會通過反射或者cglib來生成bean執行個體這都是耗性能的操作,其次給對象配置設定記憶體也會涉及複雜算法。

2.減少jvm垃圾回收

由于不會給每個請求都新生成bean執行個體,是以自然回收的對象少了。

3.可以快速擷取到bean

因為單例的擷取bean操作除了第一次生成之外其餘的都是從緩存裡擷取的是以很快。

單例bean的劣勢

單例的bean一個很大的劣勢就是他不能做到線程安全!!!

由于所有請求都共享一個bean執行個體,是以這個bean要是有狀态的一個bean的話可能在并發場景下出現問題,而原型的bean則不會有這樣問題(但也有例外,比如他被單例bean依賴),因為給每個請求都新建立執行個體。

總結

Spring 為啥把bean預設設計成單例?

答案:為了提高性能!!!

從幾個方面:

少建立執行個體

垃圾回收

緩存快速擷取

單例有啥劣勢?

如果是有狀态的話在并發環境下線程不安全。