Spring中Bean的作用域

默认情况下,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配置元素,他会告诉Springbean创建一个作用域代理。默认情况下,使用CGLib创建目标类的代理。

如果将proxy-target-class属性设置为false,则会生成基于接口的代理。

<bean id="cart" class="com.happy.learn.model.ShoppingCart" scope="session">
    <aop:scoped-proxy proxy-target-class="false"/>
</bean>
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值