• 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>
  • 相关阅读:
    Python:如何将文件映射到内存
    关于系统中:/dev/mem
    Python:如何设置文件的缓冲
    Python:如何处理二进制文件
    Python:struct模块的pack、unpack
    Python:如何读写文本文件
    《鸟哥的Linux私房菜》读书笔记3
    《鸟哥的Linux私房菜》读书笔记2
    《鸟哥的Linux私房菜》读书笔记1
    raspberry是个什么玩意
  • 原文地址:https://www.cnblogs.com/yw0219/p/5990911.html
Copyright © 2020-2023  润新知