熟悉Spring開發(fā)的朋友都知道Spring提供了5種scope分別是singleton、prototype、request、session、global session。 如下圖是官方文檔上的截圖,感興趣的朋友可以進(jìn)去看看這五種分別有什么不同。今天要介紹的是這五種中的前兩種,也是Spring最初提供的bean scope singleton 和 prototype。 Spring官方文檔介紹如下圖: 更多內(nèi)容可以看官方文檔介紹,非常詳細(xì):
單例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)求來的時(shí)候會(huì)先從緩存(map)里查看有沒有,有的話直接使用這個(gè)對(duì)象,沒有的話才實(shí)例化一個(gè)新的對(duì)象,所以這是個(gè)單例的。但是對(duì)于原型(prototype)bean來說當(dāng)每次請(qǐng)求來的時(shí)候直接實(shí)例化新的bean,沒有緩存以及從緩存查的過程。 1.畫圖分析2.源碼分析生成bean時(shí)先判斷單例的還是原型的 如果是單例的則先嘗試從緩存里獲取,沒有在新創(chuàng)建 結(jié)論:
單例bean的優(yōu)勢(shì)由于不會(huì)每次都新創(chuàng)建新對(duì)象所以有一下幾個(gè)性能上的優(yōu)勢(shì): 1.減少了新生成實(shí)例的消耗新生成實(shí)例消耗包括兩方面,第一,spring會(huì)通過反射或者cglib來生成bean實(shí)例這都是耗性能的操作,其次給對(duì)象分配內(nèi)存也會(huì)涉及復(fù)雜算法。 2.減少jvm垃圾回收由于不會(huì)給每個(gè)請(qǐng)求都新生成bean實(shí)例,所以自然回收的對(duì)象少了。 3.可以快速獲取到bean因?yàn)閱卫墨@取bean操作除了第一次生成之外其余的都是從緩存里獲取的所以很快。 有關(guān)bean實(shí)例化相關(guān)可以看著篇文章:
單例bean的劣勢(shì)單例的bean一個(gè)很大的劣勢(shì)就是他不能做到線程安全!??!,由于所有請(qǐng)求都共享一個(gè)bean實(shí)例,所以這個(gè)bean要是有狀態(tài)的一個(gè)bean的話可能在并發(fā)場(chǎng)景下出現(xiàn)問題,而原型的bean則不會(huì)有這樣問題(但也有例外,比如他被單例bean依賴),因?yàn)榻o每個(gè)請(qǐng)求都新創(chuàng)建實(shí)例。關(guān)于這方面我正在準(zhǔn)備寫一篇文章,在整理當(dāng)中,感興趣的朋友可以關(guān)注我,我后續(xù)寫一篇詳細(xì)的文章。 總結(jié)Spring 為啥把bean默認(rèn)設(shè)計(jì)成單例?答案:為了提高性能?。?!從幾個(gè)方面,
單例有啥劣勢(shì)?如果是有狀態(tài)的話在并發(fā)環(huán)境下線程不安全 END |
|