Spring定义了多种作用域,我们在使用的时候可以根据使用的需求来选择对应的作用域,这些作用域,包括(第二个括号中为更安全的注解方法,具体更多参数可查看接口代码)
- 单例(Singleton)(ConfigurableBeanFactory.SCOPE_SINGLETON):在整个应用中,只创建bean的一个实例。
- 原型(Prototype)(ConfigurableBeanFactory.SCOPE_PROTOTYPE):每次注入或者通过Spring应用上下文获取的时候,都会创建一个新的bean实例。
- 会话(Session)(WebApplicationContext.SCOPE_SESSION):在Web应用中,为每个会话创建一个bean实例。
- 请求(Rquest)(WebApplicationContext.SCOPE_REQUEST):在Web应用中,为每个请求创建一个bean实例。
单例是默认的作用域,但是正如之前所述,对于易变的类型,这并不合适。如果选择其他的作用域,要使用@Scope注解,它可以与@Component或@Bean一起使用。
我们可以分别使用三种配置Bean的方法来指定Bean的作用域。
使用隐式的组件扫描方式:
@Component @Scope("prototype") //@Scope(ConfigurableBeanFactory.SCOPE_PROTOTYPE) public class DangNianQing implements CompactDisc { } }
使用java方式配置
@Bean @Scope("prototype") public CDPlay cdPlay(CompactDisc compactDisc){ return new CDPlay(compactDisc); }
使用xml方式配置
<bean id="dangNianQing" class="com.bean.xml.DangNianQing" scope="prototype"/>
假如我们现在正在做一个购物的网站,就购物车这个Bean来说,如果没有指定他的作用于的话,那么就是默认的单例模式,那么就会发生:所有的用户都会向同一个购物车中添加商品。另一方面,如果购物车是原型作用域的,那么在应用中某一个地方往购物车中添加商品,在应用的另外一个地方可能就不可用了,因为在这里注入的是另外一个原型作用域的购物车。所以这个时候,最适合的就是用会话作用域。
使用的方法是:
@Component @Scope(value = WebApplicationContext.SCOPE_SESSION, proxyMode = ScopedProxyMode.INTERFACES) public ShoppingCar cart(){ }
正好有个Service,有个Setter方法需要注入Bean
@Component public class storeService{ @Autowired public void setShoppingCar(ShoppingCar shoppingCar){ this.shoppingCar = shoppingCar; } }
从上面的代码中我们可以看出storeService是默认的 为单例Bean,会在Spring应用上下文加载的时候创建,当然在创建的时候也会企图去将ShoppingCar注入进来,但是ShoppingCart bean是会话作用域的,此时并不存在。直到某个用户进入系统,创建了会话之后,才会出现ShoppingCart实例。另外,系统中将会有多个ShoppingCart实例:每个用户一个。我们并不想让Spring注入某个固定的ShoppingCart实例到StoreService中。我们希望的是当StoreService处理购物车功能时,它所使用的ShoppingCart实例恰好是当前会话所对应的那一个。
但是在这个例子当中Spring并不会将实际的ShoppingCart bean注入到StoreService中,Spring会注入一个到ShoppingCart bean的代理,如下图所示。这个代理会暴露与ShoppingCart相同的方法,所以StoreService会认为它就是一个购物车。但是,当StoreService调用ShoppingCart的方法时,代理会对其进行懒解析并将调用委托给会话作用域内真正的ShoppingCart bean。
现在,我们带着对这个作用域的理解,讨论一下proxyMode属性。如配置所示,proxyMode属性被设置成了copedProxyMode.INTERFACES,这表明这个代理要实现ShoppingCart接口,并将调用委托给实现bean。
如果ShoppingCart是接口而不是类的话,这是可以的(也是最为理想的代理模式)。但如果ShoppingCart是一个具体的类的话,Spring就没有办法创建基于接口的代理了。此时,它必须使用CGLib来生成基于类的代理。所以,如果bean类型是具体类的话,我们必须要将proxyMode属性设置为ScopedProxyMode.TARGET_CLASS,以此来表明要以生成目标类扩展的方式创建代理。
下面也学习下在xml中配置作用域的代理
<bean id="dangNianQing" class="com.bean.xml.DangNianQing" scope="session"> <aop:scoped-proxy></aop:scoped-proxy> </bean>
其中的<aop:scoped-proxy>与@Scope中的proxyMode 一样的,因为Bean类型是类 不是接口所以要把 proxy-target-class属性设置为false
<bean id="dangNianQing" class="com.bean.xml.DangNianQing" scope="session"> <aop:scoped-proxy proxy-target-class="false" /> </bean>
总的来说:就是如果我们有一个接口或者类 需要注入Bean的时候,我们可以委托给代理,这样就每次请求会话的时候,都有一个与其对应的Bean了
以上就是Bean作用域相关的知识,如果有错误,请指出,谢谢