Spring-bean的作用域

在大多数情况下,单例bean是很理想的方案。初始化和垃圾回收对象实例所带来的的成本只留给一些小规模任务,在这些任务中,让对象保持无状态并且在应用中反复重用这些对象可能并不合理。在这种情况下,将class声明为单例的bean会被污染,稍后重用的时候会出现意想不到的问题。

Spring定义了多种作用域,可以基于这些作用域创建bean,包括:

  • 单例(singleton):在整个应用中,只创建bean的一个实例。
  • 原型(prototype):每次注入或者通过Spring应用上下文获取的时候,都会创建一个新的实例。
  • 会话(session):在Web应用中,为每个会话创建一个bean实例。
  • 请求(request):在Web应用中,为每个请求创建一个bean实例。

单例是默认的作用域,如果选择其他作用域,要使用@Scope注解,它可以与@Component或@Bean一起使用:

1 import org.springframework.beans.factory.config.ConfigurableBeanFactory;
2 import org.springframework.context.annotation.Scope;
3 
4 @Component
5 @Scope(ConfigurableBeanFactory.SCOPE_PROTOTYPE)
6 public class Notepad{...}

当然也可以在@Bean下使用,参数可以直接写字符串prototype,singleton等,但并没有上述方式安全。

在xml模式中,可以在bean的属性中添加scope="prototype":

<bean id="notepad" class="..."  scope="prototype"/>

使用会话和请求作用域

使用会话作用域的bean最合适的莫过于购物车了,一个用户用一个购物车,而不是共用,或者因商品种类不同而创造无用的购物车。因为它与给定的用户关联性最大。示例如下:

 1 package soundSystem;
 2 
 3 import org.springframework.context.annotation.Scope;
 4 import org.springframework.context.annotation.ScopedProxyMode;
 5 import org.springframework.stereotype.Component;
 6 import org.springframework.web.context.WebApplicationContext;
 7 
 8 @Component
 9 @Scope(value=WebApplicationContext.SCOPE_SESSION,proxyMode=ScopedProxyMode.INTERFACES)
10 public class JDShoppingCart implements ShoppingCart{
11     public ShoppingCart getJDShoppingCart(){
12         return new JDShoppingCart();
13     }
14 }

 这里,我们将value设置成了WebApplicationContext中的SCOPE_SESSION常量(它的值是session)。表明为Web应用中的每个会话创建一个ShoppingCart。这会创建多个ShoppingCart实例,但对于给定的会话,只创建一个实例。也就是说,在当前会话相关的操作中,ShoppingCart是单例的。

要注意的是,还有一个代理模式的属性,它被设置成ScopedProxyMode.INTERFACES。这个属性解决了讲会话或请求作用域的bean注入到单例bean中所遇到的问题。在描述proxyMode属性之前,我们先来看看proxyMode所解决问题的场景。假如要将ShoppingCart bean注入到单例StoreService bean的Setter方法中:

 1 package soundSystem;
 2 
 3 import org.springframework.beans.factory.annotation.Autowired;
 4 import org.springframework.stereotype.Component;
 5 
 6 @Component
 7 public class StoreService {
 8 private ShoppingCart shoppingCart;
 9 
10 @Autowired
11 public void setShoppingCart(ShoppingCart shoppingCart) {
12     this.shoppingCart = shoppingCart;
13 }
14 }

 因为StoreService是一个单例的bean,会在Spring应用上下文加载的时候创建。当它创建的时候,Spring会试图将ShoppingCart bean注入到setter方法中,但是ShoppingCart bean是会话作用域的,此时并不存在,直到某个用户进入系统,创建了会话之后,才会出现ShoppingCart实例。

另外,系统中将会有多个ShoppingCart实例:每个用户一个。我们并不想让Spring注入某个固定的ShoppingCart实例到StoreService中。我们希望的是当StoreService处理购物车功能时,它所使用的ShoppingCart实例恰好是当前会话所对应的那个。

所以Spring并不会将实际的ShoppingCart bean注入到StoreService中,Spring会注入到一个ShoppingCart bean的代理,这个代理会暴露与ShoppingCart相同的方法,所以StoreService会认为它是一个购物车。但是当StoreService调用ShoppingCart的方法时,代理会对其进行懒解析并将调用委托给会话作用域内真正的ShoppingCart bean。

现在,proxyMode被设置成ScopedProxyMode.INTERFACE常量,表明这个代理要实现ShoppingCart接口,并将调用委托给实现bean。如果需要实现类,则设置为ScopedProxyMode.TARGET_CLASS即可。

xml模式下配置代理模式

1 <bean id="jDShoppingCart" class="soundSystem.JDShoppingCart" scope="session">
2     <aop:scoped-proxy/>
3 </bean>

 

也可以将proxy-target-class设置为false,从而要求它生成基于接口的代理:

1         <bean id="jDShoppingCart" class="soundSystem.JDShoppingCart" scope="session">
2             <aop:scoped-proxy proxy-target-class="false"/>
3         </bean>
posted @ 2016-10-23 20:38  舒山  阅读(2785)  评论(0编辑  收藏  举报