默认情况下,Spring
容器中所有bean
都是以单例形式创建的,也就是说,不管给定的一个bean
被注入到其他bean
多少次,每次注入的都是同一个实例。
分类
Spring
定义了多种作用域,可以基于这些作用域创建bean
,主要有如下:
- singleton(单例):默认值,在整个应用中,只创建bean的一个实例,加载spring配置文件时,会创建单实例对象
- prototype(原型):每次注入或通过Spring应用上下文获取的时候,都会创建一个新的bean实例
- session(会话):在web应用中,为每个会话创建一个bean实例
- request(请求):在web应用中,为每个请求创建一个bean实例
会话和请求作用域后面我们会单独讲。
设置
组件扫描
如果使用组件扫描来发现和声明bean
,那么可以在bean
的类上使用@Scope
注解,将其声明为原型bean
,如下:
@Component
@Scope(ConfigurableBeanFactory.SCOPE_PROTOTYPE)
//@Scope("prototype")
public class Mobile {
}
作用域常量:
prototype: ConfigurableBeanFactory.SCOPE_PROTOTYPE
singleton:ConfigurableBeanFactory.SCOPE_SINGLETON
session:WebApplicationContext.SCOPE_SESSION
request:WebApplicationContext.SCOPE_REQUEST
可以直接使用prototype
字符串,但规范和安全起见,还是建议使用常量
Java配置
在Java
配置中可以组合使用@Scope
和@Bean
来指定所需的作用域,如下:
@Bean
@Scope(ConfigurableBeanFactory.SCOPE_PROTOTYPE)
public Mobile mobile(){
return new Mobile();
}
XML配置
如果使用xml
来配置bean
的话,可以使用<bean>
元素的scope
属性来设置作用域:
<bean id="mobile" class="com.happy.learn.model.Mobile" scope="prototype"/>
会话和请求作用域
使用会话和请求作用域
在Web应用中,特别是电子商务应用,可能会有一个bean
代表用户的购物车,如果购物车是单例的话,那么所有的用户都会向同一个购物车添加商品。另外,如果是原型作用域的话,那么在应用中某个地方往购物车添加商品,那么在应用的另一个地方可能就不可用了,因此,如果能实例化在请求或会话作用域的bean,就非常有意义了。
就购物车Bean
来说,会话作用域是最合适的,因为与给用的用户关联性最大,作用域设置如下:
@Component
@Scope(value = WebApplicationContext.SCOPE_SESSION,proxyMode = ScopedProxyMode.INTERFACES)
public class ShoppingCart {
}
这就会告诉Spring
为Web应用的每个会话创建一个ShoppingCart
,在当前会话中,这个bean
实际上相当于单例的。
需要注意的是,@Scope
同时设置了一个proxyMode
属性,该值为ScopedProxyMode.INTERFACES
,这个属性解决了将会话或请求作用域的bean
注入到单例bean
中的问题。
假如要将ShoppingCart bean
注入到单例StoreService bean
中,如下:
@Component
public class StoreService {
private ShoppingCart shoppingCart;
@Autowired
public void shoppingCart(ShoppingCart shoppingCart) {
this.shoppingCart = shoppingCart;
}
}
因为StoreService
是一个单实例的bean
,会在Spring
应用上下文加载时创建。当他创建的时候,Spring
会试图将ShoppingCart bean
注入到setShoppingCart
方法中。
但ShoppingCart bean
是会话作用域的,此时并不存在,只有当用户进入系统,创建会话之后,才会出现ShoppingCart
实例。
另外,系统中会有多个ShoppingCart
实例,即每个用户都有一个。我们并不想让Spring
注入某个固定的ShoppingCart
实例到StoreService中。我们希望的是当StoreService
处理购物车功能时,他所使用的ShoppingCart
实例恰好是当前会话所对应的那一个。
实际上,Spring
并不会将实际的ShoppingCart bean
注入到StoreService
中,Spring
会注入一个到ShoppingCart bean
的代理,如下图所示。这个代理会暴露于ShoppingCart
相同的方法,所以StoreService
会认为它就是一个购物车。
但是当StoreService调用ShoppingCart的方法时,代理会对齐进行懒解析并将调用委托给会话作用域内真正的ShoppingCart bean。
如之前的配置所示,proxyMode
属性被设置成ScopedProxyMode.INTERFACES
,这表明这个代理要实现ShoppingCart
接口,并将调用委托给实现bean
。
如果ShoppingCart
是接口而不是类的话,这也是最理想的代理模式。但如果ShoppingCart
是一个具体的类的话,Spring
就没有办法创建基于接口的代理了。此时,它必须使用CGLib
来生成基于类的代理。所以,如果bean
类型是具体类的话,就需要将proxyMode
属性设置为ScopedProxyMode.TARGET_CLASS
,以此来表明要以目标类扩展的方式创建代理。
请求作用域的装配问题同会话作用域,也以作用域代理的方式进行注入。
在XML中声明作用域代理
如果在XML
中声明会话或请求作用域的bean
,那就需要使用Spring aop
命名空间的一个元素,首先声明aop
命名空间:
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:aop="http://www.springframework.org/schema/aop"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd">
声明如下:
<bean id="cart" class="com.happy.learn.model.ShoppingCart" scope="session">
<aop:scoped-proxy/>
</bean>
<aop:scoped-proxy/>
是与@Scope
注解的proxyMode
属性功能相同的Spring XML
配置元素,他会告诉Spring
为bean
创建一个作用域代理。默认情况下,使用CGLib
创建目标类的代理。
如果将proxy-target-class
属性设置为false
,则会生成基于接口的代理。
<bean id="cart" class="com.happy.learn.model.ShoppingCart" scope="session">
<aop:scoped-proxy proxy-target-class="false"/>
</bean>