思考:Spring 為啥默認把 Bean 設計成單例的?

  • 單例bean與原型bean的區別
  • 1.畫圖分析
  • 2.源碼分析
  • 單例bean的優勢
  • 1.減少了新生成實例的消耗
  • 2.減少jvm垃圾回收
  • 3.可以快速獲取到bean
  • 單例bean的劣勢
  • 總結
  • 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 設計成單例的?

結論:

  1. 單例的bean只有第一次創建新的bean 後面都會複用該bean,所以不會頻繁創建對象。
  2. 原型的bean每次都會新創建


單例bean的優勢

由於不會每次都新創建新對象所以有一下幾個性能上的優勢:

1.減少了新生成實例的消耗

新生成實例消耗包括兩方面,第一,spring會通過反射或者cglib來生成bean實例這都是耗性能的操作,其次給對象分配內存也會涉及複雜算法。

2.減少jvm垃圾回收

由於不會給每個請求都新生成bean實例,所以自然回收的對象少了。

3.可以快速獲取到bean

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

單例bean的劣勢

單例的bean一個很大的劣勢就是他不能做到線程安全!!!,由於所有請求都共享一個bean實例,所以這個bean要是有狀態的一個bean的話可能在併發場景下出現問題,而原型的bean則不會有這樣問題(但也有例外,比如他被單例bean依賴),因為給每個請求都新創建實例。關於這方面我正在準備寫一篇文章,在整理當中,感興趣的朋友可以關注我,我後續寫一篇詳細的文章。

總結

Spring 為啥把bean默認設計成單例?

答案:為了提高性能!!!從幾個方面,

  1. 少創建實例
  2. 垃圾回收
  3. 緩存快速獲取

單例有啥劣勢?

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

來源:http://rrd.me/ekVeP
作者:清歡ysy


:-D 搜索微信號(ID:芋道源碼),可以獲得各種 Java 源碼解析、原理講解、面試題、學習指南。

:-D 並且,回覆【書籍】後,可以領取筆者推薦的各種 Java 從入門到架構的 100 本書籍。

:-D 並且,回覆【技術群】後,可以加入專門討論 Java、後端、架構的技術群。

思考:Spring 為啥默認把 Bean 設計成單例的?

來吧,騷年~


分享到:


相關文章: