spring bean scope 用法分析

简介: Spring 容器是应用组件 bean 创建和组装专家, 支持 `InitializingBean`, `AbstractBeanDefinition.setInitMethodName(String)`, `@PostConstruct` 等多种方法初始化(及销毁)bean。 ## 单例(singleton)bean 最常使用的是单例(singleton)bean,Spring 自

Spring 容器是应用组件 bean 创建和组装专家,
支持 InitializingBean, AbstractBeanDefinition.setInitMethodName(String), @PostConstruct 等多种方法初始化(及销毁)bean。

单例(singleton)bean

最常使用的是单例(singleton)bean,Spring 自动帮我们管理其生命周期。

代码示例如下:

class Rick {
    @PostConstruct
    void init() {
        System.out.println("rick init");
    }
    @PreDestroy
    void destroy() {
        System.out.println("rick destroy");
    }
}

@Configuration
class RickConfig {
    @Bean
    Rick rick() {
        return new Rick();
    }
}

@Configuration
@ComponentScan
public class App {
    public static void main(String[] args) {
        SpringApplication.run(App.class, args);
    }
}

运行结果如下:

rick init
rick destroy

原型(prototype)bean

有些场景不适合使用单例 bean,因此 spring 还支持原型(prototype)bean,每次请求 bean 都返回一个新的实例。

一段 错误的 代码示例如下,每个平行宇宙都有一个 rick:

class Rick {
    static int NextId;
    int id = NextId++;
    @PostConstruct
    void init() {
        System.out.println("rick init " + id);
    }
    @PreDestroy
    void destroy() {
        System.out.println("rick destroy " + id);
    }
}

@Configuration
class RickConfig {
    @Bean
    @Scope(ConfigurableBeanFactory.SCOPE_PROTOTYPE)
    Rick rickPrototype() {
        return new Rick();
    }
    @Lookup
    Rick rick() {
        throw new IllegalStateException("this method would be override by spring.");
    }
}

@Component
class WorldC37 {
    @Autowired
    Rick rick;
}

@Component
class WorldZ {
    @Autowired
    Rick rick;
}

@Configuration
@ComponentScan
public class App implements ApplicationRunner {
    public static void main(String[] args) {
        SpringApplication.run(App.class, args);
    }
    @Autowired
    RickConfig rickConfig;
    @Override
    public void run(ApplicationArguments args) throws Exception {
        Rick rick = rickConfig.rick();
    }
}

运行结果如下:

rick init 0
rick init 1
rick init 2

问题:

  1. 很显然,原型 bean 被实例化了多次,但没有被销毁。

    原型 bean 很灵活,除了容器初始化时创建外,应用运行时还可以通过 `@Lookup` 方法任意创建。

    因此 spring 采取了放养政策,初始化完就交给应用自己处理,不会自动管理原型 bean 的生命周期。

这段代码还有其他不足:

  1. 每个平行宇宙的 rick 可能很不一样,需要支持个性化定制。
  2. 一个宇宙只有一个 rick,动态创建应该跟其宇宙绑定,而不是总是不停的创建。

先来解第一个问题:Spring 让应用自己管理原型 bean,前面提到 bean 支持多种销毁方法,如何保证处理正确?
靠谱的做法还是交回 Spring 销毁,但应用负责决定什么时候 bean 不要了,应该销毁,回调 Spring。
原型 bean 可动态创建,也可以动态销毁,所以应用可以更灵活的控制其生命周期。
适时调用 ConfigurableBeanFactory.destroyBean(String beanName, Object beanInstance) 方法即可销毁 bean。

订正后的代码示例如下:

abstract class AbstractWorld extends ApplicationObjectSupport {
    @Autowired
    Rick rick;
    @PreDestroy
    void destroy() {
        ((ConfigurableApplicationContext) getApplicationContext()).getBeanFactory().destroyBean("rickPrototype", rick);
    }
}

@Component
class WorldC37 extends AbstractWorld {
}

@Component
class WorldZ extends AbstractWorld {
}

@Configuration
@ComponentScan
public class App extends ApplicationObjectSupport implements ApplicationRunner {
    public static void main(String[] args) {
        SpringApplication.run(App.class, args);
    }
    @Autowired
    RickConfig rickConfig;
    @Override
    public void run(ApplicationArguments args) throws Exception {
        Rick rick = rickConfig.rick();
        ((ConfigurableApplicationContext) getApplicationContext()).getBeanFactory().destroyBean("rickPrototype", rick);
        rick = null;
    }
}

运行结果:

rick init 0
rick init 1
rick init 2
rick destroy 2
rick destroy 1
rick destroy 0

可看到原型 bean 被正确销毁。

自定义 scope

Spring 还支持自定义 scope,其用法介于单例和原型之间。

如之前说过的,每个新的平行宇宙会有新的一个 rick,但同一个宇宙总会获取到同一个 rick(而不是总是新建),这里宇宙即可作为一种自定义 scope。
详情参考文档:https://docs.spring.io/spring/docs/4.3.13.RELEASE/spring-framework-reference/htmlsingle/#beans-factory-scopes-custom
此处不再细说。

相关文章
|
4天前
|
XML Java 数据格式
Spring5入门到实战------7、IOC容器-Bean管理XML方式(外部属性文件)
这篇文章是Spring5框架的实战教程,主要介绍了如何在Spring的IOC容器中通过XML配置方式使用外部属性文件来管理Bean,特别是数据库连接池的配置。文章详细讲解了创建属性文件、引入属性文件到Spring配置、以及如何使用属性占位符来引用属性文件中的值。
Spring5入门到实战------7、IOC容器-Bean管理XML方式(外部属性文件)
|
4天前
|
XML Java 数据格式
Spring5入门到实战------4、IOC容器-Bean管理XML方式、集合的注入(二)
这篇文章是Spring5框架的实战教程,主题是IOC容器中Bean的集合属性注入,通过XML配置方式。文章详细讲解了如何在Spring中注入数组、List、Map和Set类型的集合属性,并提供了相应的XML配置示例和Java类定义。此外,还介绍了如何在集合中注入对象类型值,以及如何使用Spring的util命名空间来实现集合的复用。最后,通过测试代码和结果展示了注入效果。
Spring5入门到实战------4、IOC容器-Bean管理XML方式、集合的注入(二)
|
4天前
|
XML Java 数据格式
Spring5入门到实战------6、IOC容器-Bean管理XML方式(自动装配)
这篇文章是Spring5框架的入门教程,详细讲解了IOC容器中Bean的自动装配机制,包括手动装配、`byName`和`byType`两种自动装配方式,并通过XML配置文件和Java代码示例展示了如何在Spring中实现自动装配。
Spring5入门到实战------6、IOC容器-Bean管理XML方式(自动装配)
|
4天前
|
XML Java 数据格式
Spring5入门到实战------5、IOC容器-Bean管理(三)
这篇文章深入探讨了Spring5框架中IOC容器的高级Bean管理,包括FactoryBean的使用、Bean作用域的设置、Bean生命周期的详细解释以及Bean后置处理器的实现和应用。
Spring5入门到实战------5、IOC容器-Bean管理(三)
|
4天前
|
XML Java 数据格式
Spring5入门到实战------3、IOC容器-Bean管理XML方式(一)
这篇文章详细介绍了Spring框架中IOC容器的Bean管理,特别是基于XML配置方式的实现。文章涵盖了Bean的定义、属性注入、使用set方法和构造函数注入,以及如何注入不同类型的属性,包括null值、特殊字符和外部bean。此外,还探讨了内部bean的概念及其与外部bean的比较,并提供了相应的示例代码和测试结果。
Spring5入门到实战------3、IOC容器-Bean管理XML方式(一)
|
4天前
|
XML Java 数据格式
Spring5入门到实战------8、IOC容器-Bean管理注解方式
这篇文章详细介绍了Spring5框架中使用注解进行Bean管理的方法,包括创建Bean的注解、自动装配和属性注入的注解,以及如何用配置类替代XML配置文件实现完全注解开发。
Spring5入门到实战------8、IOC容器-Bean管理注解方式
|
22天前
|
Java Spring 容器
Spring Boot 启动源码解析结合Spring Bean生命周期分析
Spring Boot 启动源码解析结合Spring Bean生命周期分析
60 11
|
23天前
|
消息中间件 Java Kafka
Spring boot 自定义kafkaTemplate的bean实例进行生产消息和发送消息
Spring boot 自定义kafkaTemplate的bean实例进行生产消息和发送消息
34 5
|
16天前
|
Java Spring
Spring的Bean生命周期中@PostConstruct注解
【8月更文挑战第3天】在Spring框架中,`@PostConstruct`注解标示Bean初始化完成后立即执行的方法。它在依赖注入完成后调用,适用于资源加载、属性设置等初始化操作。若方法中抛出异常,可能影响Bean初始化。与之对应,`@PreDestroy`注解的方法则在Bean销毁前执行,用于资源释放。
|
21天前
|
IDE Java 开发工具
解决非Spring Bean访问Spring Bean的问题:实用指南
在非SpringBean类中直接获取SpringBean可能会引发问题,例如上面案例里提到的空指针和自动装配失败。为避免这些问题,建议将需要访问Spring Bean的类也注册为Spring Bean,以确保依赖关系得到正确管理。
15 0