3.4 bean 的作用域

在默认情况下,Spring 应用上下文中所有 bean 都是作为以单例(singleton)的形式创建的。也就是说,不管给定的一个 bean 被注入到其他 bean 多少次,每次所注入的都是同一个实例。

在大多数情况下,单例 bean 是很理想的方案。初始化和垃圾回收对象实例所带来的成本只留给一些小规模任务,在这些任务中,让对象保持无状态并且在应用中反复重用这些对象可能并不合理。

有时候,可能会发现,你所使用的类是易变的(mutable),它们会保持一些状态,因此重用是不安全的。在这种情况下,将 class 声明为单例的 bean 就不是什么好主意了,因为对象会被污染,稍后重用的时候会出现意想不到的问题。

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

  • 单例(Singleton):在整个应用中,只创建 bean 的一个实例。

  • 原型(Prototype):每次注入或者通过 Spring 应用上下文获取的时候,都会创建一个新的 bean 实例。

  • 会话(Session):在 Web 应用中,为每个会话创建一个 bean 实例。

  • 请求(Rquest):在 Web 应用中,为每个请求创建一个 bean 实例。

单例是默认的作用域,但是正如之前所述,对于易变的类型,这并不合适。如果选择其他的作用域,要使用 @Scope 注解,它可以与 @Component 或 @Bean 一起使用。

例如,如果你使用组件扫描来发现和声明 bean,那么你可以在 bean 的类上使用 @Scope 注解,将其声明为原型 bean:

Notepad.java
@Component
@Scop(ConfigurableBeanFactory.SCOPE_PROTOTYPE)
public class Notepad { ... }

这里,使用 ConfigurableBeanFactory 类的 SCOPE_PROTOTYPE 常量设置了原型作用域。你当然也可以使用 @Scope("prototype"),但是使用 SCOPE_PROTOTYPE 常量更加安全并且不易出错。

如果你想在 Java 配置中将 Notepad 声明为原型 bean,那么可以组合使用 @Scope 和 @Bean 来指定所需的作用域:

Notepad.java
@Bean
@Scope(ConfigurableBeanFactory.SCOPE_PROTOTYPE)
public Notepad notepad() {
return new Notepad();
}

同样,如果你使用 XML 来配置 bean 的话,可以使用元素的 scope 属性来设置作用域:

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

不管你使用哪种方式来声明原型作用域,每次注入或从 Spring 应用上下文中检索该 bean 的时候,都会创建新的实例。这样所导致的结果就是每次操作都能得到自己的 Notepad 实例。