天天看點

詳解Spring中bean的作用域

<bean id="role" class="spring.chapter2.maryGame.Role" scope="singleton"/>
      

這裡的scope就是用來配置spring bean的作用域,它辨別bean的作用域。

在spring2.0之前bean隻有2種作用域即:singleton(單例)、non-singleton(也稱prototype),Spring2.0以後,增加了session、request、global session三種專用于Web應用程式上下文的Bean。是以,預設情況下Spring2.0現在有五種類型的Bean。當然,Spring2.0對Bean的類型的設計進行了重構,并設計出靈活的Bean類型支援,理論上可以有無數多種類型的Bean,使用者可以根據自己的需要,增加新的Bean類型,滿足實際應用需求。

1、singleton作用域

當一個bean的作用域設定為singleton,那麼Spring IOC容器中隻會存在一個共享的bean執行個體,并且所有對bean的請求,隻要id與該bean定義相比對,則隻會傳回bean的同一執行個體。換言之,當把一個bean定義設定為singleton作用域時,Spring IOC容器隻會建立該bean定義的唯一執行個體。這個單一執行個體會被存儲到單例緩存(singleton cache)中,并且所有針對該bean的後續請求和引用都将傳回被緩存的對象執行個體,這裡要注意的是singleton作用域和GOF設計模式中的單例是完全不同的,單例設計模式表示一個ClassLoader中隻有一個class存在,而這裡的singleton則表示一個容器對應一個bean,也就是說當一個bean被辨別為singleton時候,spring的IOC容器中隻會存在一個該bean。

配置執行個體:

<bean id="role" class="spring.chapter2.maryGame.Role" scope="singleton"/>
      

或者

<bean id="role" class="spring.chapter2.maryGame.Role" singleton="true"/>
      

2、prototype

prototype作用域部署的bean,每一次請求(将其注入到另一個bean中,或者以程式的方式調用容器的getBean()方法)都會産生一個新的bean執行個體,相當于一個new的操作,對于prototype作用域的bean,有一點非常重要,那就是Spring不能對一個prototype bean的整個生命周期負責,容器在初始化、配置、裝飾或者是裝配完一個prototype執行個體後,将它交給用戶端,随後就對該prototype執行個體不聞不問了。不管何種作用域,容器都會調用所有對象的初始化生命周期回調方法,而對prototype而言,任何配置好的析構生命周期回調方法都将不會被調用。清除prototype作用域的對象并釋放任何prototype bean所持有的昂貴資源,都是用戶端代碼的職責。(讓Spring容器釋放被singleton作用域bean占用資源的一種可行方式是,通過使用bean的後置處理器,該處理器持有要被清除的bean的引用。)

配置執行個體:

<bean id="role" class="spring.chapter2.maryGame.Role" scope="prototype"/>
      

或者

<beanid="role" class="spring.chapter2.maryGame.Role" singleton="false"/>
      

3、request

request表示該針對每一次HTTP請求都會産生一個新的bean,同時該bean僅在目前HTTP request内有效,配置執行個體:

request、session、global session使用的時候,首先要在初始化web的web.xml中做如下配置:

如果你使用的是Servlet 2.4及以上的web容器,那麼你僅需要在web應用的XML聲明檔案web.xml中增加下述ContextListener即可:

<web-app>
...
<listener>
<listener-class>org.springframework.web.context.request.RequestContextListener</listener-class>
</listener>
...
</web-app>
      

如果是Servlet2.4以前的web容器,那麼你要使用一個javax.servlet.Filter的實作:

<web-app>
..
<filter> 
<filter-name>requestContextFilter</filter-name> 
<filter-class>org.springframework.web.filter.RequestContextFilter</filter-class>
</filter> 
<filter-mapping> 
<filter-name>requestContextFilter</filter-name> 
<url-pattern>/*</url-pattern>
</filter-mapping>
...
</web-app>
      

接着既可以配置bean的作用域了:

<bean id="role" class="spring.chapter2.maryGame.Role" scope="request"/>
      

4、session

session作用域表示該針對每一次HTTP請求都會産生一個新的bean,同時該bean僅在目前HTTP session内有效,配置執行個體:

配置執行個體:

和request配置執行個體的前提一樣,配置好web啟動檔案就可以如下配置:

<bean id="role" class="spring.chapter2.maryGame.Role" scope="session"/>
      

5、global session

global session作用域類似于标準的HTTP Session作用域,不過它僅僅在基于portlet的web應用中才有意義。Portlet規範定義了全局Session的概念,它被所有構成某個portlet web應用的各種不同的portlet所共享。在global session作用域中定義的bean被限定于全局portlet Session的生命周期範圍内。如果你在web中使用global session作用域來辨別bean,那麼,web會自動當成session類型來使用。

配置執行個體:

和request配置執行個體的前提一樣,配置好web啟動檔案就可以如下配置:

<bean id="role" class="spring.chapter2.maryGame.Role" scope="global session"/>
      

6、自定義bean裝配作用域

在spring 2.0中作用域是可以任意擴充的,你可以自定義作用域,甚至你也可以重新定義已有的作用域(但是你不能覆寫singleton和prototype),spring的作用域由接口org.springframework.beans.factory.config.Scope來定義,自定義自己的作用域隻要實作該接口即可,下面給個執行個體:

我們建立一個線程的scope,該scope在表示一個線程中有效,代碼如下:

publicclass MyScope implements Scope ...{ 
privatefinal ThreadLocal threadScope = new ThreadLocal() ...{
protected Object initialValue() ...{
returnnew HashMap(); 
} 
}; 
public Object get(String name, ObjectFactory objectFactory) ...{ 
Map scope = (Map) threadScope.get(); 
Object object = scope.get(name); 
if(object==null) ...{ 
object = objectFactory.getObject(); 
scope.put(name, object); 
} 
return object; 
} 
public Object remove(String name) ...{ 
Map scope = (Map) threadScope.get(); 
return scope.remove(name); 
}
publicvoid registerDestructionCallback(String name, Runnable callback) ...{ 
}
public String getConversationId() ...{
// TODO Auto-generated method stub
returnnull;
} 
}