Spring bean為什么默認(rèn)是單例
熟悉Spring開(kāi)發(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ū)別
如果一個(gè)bean被聲明為單例的時(shí)候,在處理多次請(qǐng)求的時(shí)候在Spring容器里只實(shí)例化出一個(gè)bean,后續(xù)的請(qǐng)求都公用這個(gè)對(duì)象,這個(gè)對(duì)象會(huì)保存在一個(gè)map里面。當(dāng)有請(qǐng)求來(lái)的時(shí)候會(huì)先從緩存(map)里查看有沒(méi)有,有的話直接使用這個(gè)對(duì)象,沒(méi)有的話才實(shí)例化一個(gè)新的對(duì)象,所以這是個(gè)單例的。但是對(duì)于原型(prototype)bean來(lái)說(shuō)當(dāng)每次請(qǐng)求來(lái)的時(shí)候直接實(shí)例化新的bean,沒(méi)有緩存以及從緩存查的過(guò)程。
1、畫(huà)圖分析


2、源碼分析
生成bean時(shí)先判斷單例的還是原型的

如果是單例的則先嘗試從緩存里獲取,沒(méi)有在新創(chuàng)建

結(jié)論:
單例的bean只有第一次創(chuàng)建新的bean 后面都會(huì)復(fù)用該bean,所以不會(huì)頻繁創(chuàng)建對(duì)象原型的bean每次都會(huì)新創(chuàng)建
二、單例bean的優(yōu)勢(shì)
由于不會(huì)每次都新創(chuàng)建新對(duì)象所以有一下幾個(gè)性能上的優(yōu)勢(shì)。
1、減少了新生成實(shí)例的消耗
新生成實(shí)例消耗包括兩方面,第一,spring會(huì)通過(guò)反射或者cglib來(lái)生成bean實(shí)例這都是耗性能的操作,其次給對(duì)象分配內(nèi)存也會(huì)涉及復(fù)雜算法。
2、減少jvm垃圾回收
由于不會(huì)給每個(gè)請(qǐng)求都新生成bean實(shí)例,所以自然回收的對(duì)象少了。
3、可以快速獲取到bean
因?yàn)閱卫墨@取bean操作除了第一次生成之外其余的都是從緩存里獲取的所以很快。
三、單例bean的劣勢(shì)
單例的bean一個(gè)很大的劣勢(shì)就是他不能做到線程安全!由于所有請(qǐng)求都共享一個(gè)bean實(shí)例,所以這個(gè)bean要是有狀態(tài)的一個(gè)bean的話可能在并發(fā)場(chǎng)景下出現(xiàn)問(wèn)題,而原型的bean則不會(huì)有這樣問(wèn)題(但也有例外,比如他被單例bean依賴),因?yàn)榻o每個(gè)請(qǐng)求都新創(chuàng)建實(shí)例。
四、總結(jié)
1、Spring 為啥把bean默認(rèn)設(shè)計(jì)成單例?為了提高性能少創(chuàng)建實(shí)例垃圾回收緩存快速獲取
2、單例有啥劣勢(shì)?
如果是有狀態(tài)的話在并發(fā)環(huán)境下線程不安全
3、什么是有狀態(tài)對(duì)象?什么是無(wú)狀態(tài)對(duì)象?有狀態(tài)對(duì)象:有實(shí)例變量可以標(biāo)志其對(duì)象所處的狀態(tài)。(有實(shí)例變量的對(duì)象,有存儲(chǔ)數(shù)據(jù)能力)- 白話:有屬性的對(duì)象無(wú)狀態(tài)對(duì)象:無(wú)實(shí)例變量可以標(biāo)志其對(duì)象所處的狀態(tài)。(無(wú)實(shí)例變量的對(duì)象,無(wú)存儲(chǔ)數(shù)據(jù)能力)- 白話:無(wú)屬性的對(duì)象
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Java對(duì)xls文件進(jìn)行讀寫(xiě)操作示例代碼
Java開(kāi)發(fā)項(xiàng)目中經(jīng)常會(huì)碰到處理Excel文件中數(shù)據(jù)的情況,下面這篇文章主要給大家介紹了利用Java對(duì)xls文件進(jìn)行讀寫(xiě)操作的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起看看吧。2017-08-08
Java方法的參數(shù)傳遞機(jī)制實(shí)例詳解
這篇文章主要介紹了Java方法的參數(shù)傳遞機(jī)制,結(jié)合實(shí)例形式詳細(xì)分析了java方法參數(shù)傳遞機(jī)制原理、實(shí)現(xiàn)方法及操作注意事項(xiàng),需要的朋友可以參考下2019-09-09
在IDEA中創(chuàng)建跑得起來(lái)的Springboot項(xiàng)目
這篇文章主要介紹了在IDEA中創(chuàng)建跑得起來(lái)的Springboot項(xiàng)目的圖文教程,需要的朋友可以參考下2018-04-04
Java 關(guān)于時(shí)間復(fù)雜度和空間復(fù)雜度的深度刨析
算法復(fù)雜度分為時(shí)間復(fù)雜度和空間復(fù)雜度。其作用: 時(shí)間復(fù)雜度是度量算法執(zhí)行的時(shí)間長(zhǎng)短;而空間復(fù)雜度是度量算法所需存儲(chǔ)空間的大小2021-11-11
Mac系統(tǒng)搭建JDK及JMETER過(guò)程解析
這篇文章主要介紹了Mac系統(tǒng)搭建JDK及JMETER過(guò)程解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-08-08
mybatis typeAliases 給實(shí)體類起別名的方法
這篇文章主要介紹了mybatis typeAliases 給實(shí)體類起別名,本文給大家分享兩種用法,通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-09-09

