在默認(rèn)情況下,Spring應(yīng)用上下文中所有的bean都是以單例的形式創(chuàng)建的。
1.spring定義了多種作用域,可以基于這些作用域創(chuàng)建bean,包括
(1).單例:在整個(gè)應(yīng)用中,值創(chuàng)建bean的一個(gè)實(shí)例。是spring默認(rèn)的作用域。
(2).原型:每次注入或者通過(guò)spring應(yīng)用上下文獲取的時(shí)候,都會(huì)創(chuàng)建一個(gè)新的bean的實(shí)例。
(3).會(huì)話:在web應(yīng)用中,會(huì)為每個(gè)會(huì)話創(chuàng)建一個(gè)bean實(shí)例。
(4).請(qǐng)求:在web應(yīng)用中,為每個(gè)請(qǐng)求創(chuàng)建一個(gè)bean實(shí)例。
2.如果需要指定除單例外的其他作用域,需要使用@Scope注解,形式如下:
@Scope(ConfigurableBeanFactory.SCOPE_PROTOTYPE)
如果使用XML來(lái)配置的話,可以使用<bean>元素的scope屬性來(lái)設(shè)置作用域。
3.使用會(huì)話和請(qǐng)求作用域
假設(shè)我們?cè)谔詫氋?gòu)物,購(gòu)物車如果是單例的,那么所有人的商品都會(huì)被添加到同一個(gè)購(gòu)物車。如果是原型的,那么再其他的地方就不可以使用,所以此時(shí)我們可以考慮會(huì)話作用域。

再來(lái)了解一下ScopeProxyMode這個(gè)枚舉類,具體如下顯示

可以看出里面有四種選擇,分別是DEFAULT,NO,INTERFACE,TARGET_CLASS。
我們?cè)賮?lái)看一下WebApplicationContext接口,如下圖所示:

從上面我們可看出有多種SCOPE的作用域,而且WebApplicationContext接口繼承了ApplicationContext接口,下圖是ApplicationContext接口的定義:

Spring是沒(méi)有辦法創(chuàng)建基于接口的代理。它必須使用CGLib來(lái)生成基于類的代理。所以,如果bean類型是具體類的話,我們必須要將proxyMode屬性設(shè)置為ScopeProxyMode.TARGET_CLASS,以此來(lái)表明要以生成目標(biāo)類擴(kuò)展的方式創(chuàng)建代理。
上面看了一下bean作用域的注解方式,接下來(lái)看看bean的作用域的xml配置方式:
(1)<bean>元素有scope屬性,通過(guò)scope來(lái)配置bean的作用域。
(2)spring會(huì)為bean創(chuàng)建一個(gè)作用域代理,默認(rèn)情況下,它會(huì)使用CGLib創(chuàng)建目標(biāo)類的代理。也可以通過(guò)設(shè)置proxy-target-class屬性設(shè)置為false,來(lái)生成基于接口的代理。使用<aop:scoped-proxy>,要使用這個(gè)元素,必須在XML的beans根元素中聲明該命名空間。