熱點推薦:
您现在的位置: 電腦知識網 >> 編程 >> Java編程 >> Java核心技術 >> 正文

詳解Spring中bean的scope

2022-06-13   來源: Java核心技術 

  如何使用spring的作用域

  <bean id=role class=springchaptermaryGameRole scope=singleton/>

  這裡的scope就是用來配置spring bean的作用域它標識bean的作用域在spring之前bean只有種作用域即singleton(單例)nonsingleton(也稱prototype) Spring以後增加了sessionrequestglobal session三種專用於Web應用程序上下文的Bean因此默認情況下Spring現在有五種類型的Bean當然Spring對Bean的類型的設計進行了重構並設計出靈活的Bean類型支持理論上可以有無數多種類型的Bean用戶可以根據自己的需要增加新的Bean類型滿足實際應用需求

  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=springchaptermaryGameRole scope=singleton/>

  或者

  <bean id=role class=springchaptermaryGameRole singleton=true/>

  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=springchaptermaryGameRole scope=prototype/>

  或者

  <beanid=role class=springchaptermaryGameRole singleton=false/>

  request

  request表示該針對每一次HTTP請求都會產生一個新的bean同時該bean僅在當前HTTP request內有效配置實例 requestsessionglobal session使用的時候首先要在初始化web的webxml中做如下配置如果你使用的是Servlet 及以上的web容器那麼你僅需要在web應用的XML聲明文件webxml中增加下述ContextListener即可

  <webapp>

  

  <listener>

  <listenerclass>orgsprntextrequestRequestContextListener</listenerclass>

  </listener>

  

  </webapp>

  如果是Servlet以前的web容器那麼你要使用一個javaxservletFilter的實現

  <webapp>

  

  <filter>

  <filtername>requestContextFilter</filtername>

  <filterclass>orgspringframeworkwebfilterRequestContextFilter</filterclass>

  </filter>

  <filtermapping>

  <filtername>requestContextFilter</filtername>

  <urlpattern>/*</urlpattern>

  </filtermapping>

  

  </webapp>

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

  <bean id=role class=springchaptermaryGameRole scope=request/>

  session

  session作用域表示該針對每一次HTTP請求都會產生一個新的bean同時該bean僅在當前HTTP session內有效配置實例

  配置實例和request配置實例的前提一樣配置好web啟動文件就可以如下配置

  <bean id=role class=springchaptermaryGameRole scope=session/>

  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=springchaptermaryGameRole scope=global session/>

  自定義bean裝配作用域在spring中作用域是可以任意擴展的你可以自定義作用域甚至你也可以重新定義已有的作用域(但是你不能覆蓋singleton和prototype)spring的作用域由接口orgspringframewonfigScope來定義自定義自己的作用域只要實現該接口即可下面給個實例我們建立一個線程的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) threadScopeget();

  Object object = scopeget(name);

  if(object==null) {

  object = objectFactorygetObject();

  scopeput(name object);

  }

  return object;

  }

  public Object remove(String name) {

  Map scope = (Map) threadScopeget();

  return scoperemove(name);

  }

  publicvoid registerDestructionCallback(String name Runnable callback) {

  }

  public String getConversationId() {

  // TODO Autogenerated method stub

  returnnull;

  }

  }


From:http://tw.wingwit.com/Article/program/Java/hx/201311/26260.html
  • 上一篇文章:

  • 下一篇文章:
  • 推薦文章
    Copyright © 2005-2022 電腦知識網 Computer Knowledge   All rights reserved.