javascript
想过没有,Spring的Bean为啥默认单例?
熟悉Spring開發(fā)的朋友都知道Spring提供了5種scope分別是singleton、prototype、request、session、global session。
如下圖是官方文檔上的截圖,感興趣的朋友可以進(jìn)去看看這五種分別有什么不同。今天要介紹的是這五種中的前兩種,也是Spring最初提供的bean scope singleton 和 prototype。
Spring官方文檔介紹如下圖:
更多內(nèi)容可以看官方文檔介紹,非常詳細(xì):
https://docs.spring.io/spring/docs/3.0.0.M3/reference/html/ch04s04.html
單例bean與原型bean的區(qū)別
如果一個bean被聲明為單例的時候,在處理多次請求的時候在Spring容器里只實例化出一個bean,后續(xù)的請求都公用這個對象,這個對象會保存在一個map里面。當(dāng)有請求來的時候會先從緩存(map)里查看有沒有,有的話直接使用這個對象,沒有的話才實例化一個新的對象,所以這是個單例的。但是對于原型(prototype)bean來說當(dāng)每次請求來的時候直接實例化新的bean,沒有緩存以及從緩存查的過程。
1.畫圖分析
2.源碼分析
生成bean時先判斷單例的還是原型的
如果是單例的則先嘗試從緩存里獲取,沒有在新創(chuàng)建
結(jié)論:
單例的bean只有第一次創(chuàng)建新的bean 后面都會復(fù)用該bean,所以不會頻繁創(chuàng)建對象。
原型的bean每次都會新創(chuàng)建
單例bean的優(yōu)勢
由于不會每次都新創(chuàng)建新對象所以有一下幾個性能上的優(yōu)勢:
1.減少了新生成實例的消耗
新生成實例消耗包括兩方面,第一,spring會通過反射或者cglib來生成bean實例這都是耗性能的操作,其次給對象分配內(nèi)存也會涉及復(fù)雜算法。
2.減少jvm垃圾回收
由于不會給每個請求都新生成bean實例,所以自然回收的對象少了。
3.可以快速獲取到bean
因為單例的獲取bean操作除了第一次生成之外其余的都是從緩存里獲取的所以很快。
單例bean的劣勢
單例的bean一個很大的劣勢就是他不能做到線程安全!!!,由于所有請求都共享一個bean實例,所以這個bean要是有狀態(tài)的一個bean的話可能在并發(fā)場景下出現(xiàn)問題,而原型的bean則不會有這樣問題(但也有例外,比如他被單例bean依賴),因為給每個請求都新創(chuàng)建實例。關(guān)于這方面我正在準(zhǔn)備寫一篇文章,在整理當(dāng)中,感興趣的朋友可以關(guān)注我,我后續(xù)寫一篇詳細(xì)的文章。
總結(jié)
Spring 為啥把bean默認(rèn)設(shè)計成單例?
答案:為了提高性能!!!從幾個方面,
少創(chuàng)建實例
垃圾回收
緩存快速獲取
單例有啥劣勢?
如果是有狀態(tài)的話在并發(fā)環(huán)境下線程不安全。
思考題
在這里,給大家留下一個思考題,什么情況下,我們要將Spring的bean設(shè)置成非單例的?為什么要這么做?
總結(jié)
以上是生活随笔為你收集整理的想过没有,Spring的Bean为啥默认单例?的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 阿里不让 MySQL 多表 Join ?
- 下一篇: 如何快速判断某 URL 是否在 20 亿