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
此处不再细说。

相关文章
|
10天前
|
缓存 Java Spring
实战指南:四种调整 Spring Bean 初始化顺序的方案
本文探讨了如何调整 Spring Boot 中 Bean 的初始化顺序,以满足业务需求。文章通过四种方案进行了详细分析: 1. **方案一 (@Order)**:通过 `@Order` 注解设置 Bean 的初始化顺序,但发现 `@PostConstruct` 会影响顺序。 2. **方案二 (SmartInitializingSingleton)**:在所有单例 Bean 初始化后执行额外的初始化工作,但无法精确控制特定 Bean 的顺序。 3. **方案三 (@DependsOn)**:通过 `@DependsOn` 注解指定 Bean 之间的依赖关系,成功实现顺序控制,但耦合性较高。
实战指南:四种调整 Spring Bean 初始化顺序的方案
|
1月前
|
XML Java 数据格式
Spring从入门到入土(bean的一些子标签及注解的使用)
本文详细介绍了Spring框架中Bean的创建和使用,包括使用XML配置文件中的标签和注解来创建和管理Bean,以及如何通过构造器、Setter方法和属性注入来配置Bean。
66 9
Spring从入门到入土(bean的一些子标签及注解的使用)
|
28天前
|
Java 测试技术 Windows
咦!Spring容器里为什么没有我需要的Bean?
【10月更文挑战第11天】项目经理给小菜分配了一个紧急需求,小菜迅速搭建了一个SpringBoot项目并完成了开发。然而,启动测试时发现接口404,原因是控制器包不在默认扫描路径下。通过配置`@ComponentScan`的`basePackages`字段,解决了问题。总结:`@SpringBootApplication`默认只扫描当前包下的组件,需要扫描其他包时需配置`@ComponentScan`。
|
2月前
|
XML Java 数据格式
spring复习02,xml配置管理bean
详细讲解了Spring框架中基于XML配置文件管理bean的各种方式,包括获取bean、依赖注入、特殊值处理、属性赋值、集合类型处理、p命名空间、bean作用域及生命周期和自动装配。
spring复习02,xml配置管理bean
|
1月前
|
Java BI API
spring boot 整合 itextpdf 导出 PDF,写入大文本,写入HTML代码,分析当下导出PDF的几个工具
这篇文章介绍了如何在Spring Boot项目中整合iTextPDF库来导出PDF文件,包括写入大文本和HTML代码,并分析了几种常用的Java PDF导出工具。
404 0
spring boot 整合 itextpdf 导出 PDF,写入大文本,写入HTML代码,分析当下导出PDF的几个工具
|
1月前
|
Java 开发者 Spring
Spring bean的生命周期详解!
本文详细解析Spring Bean的生命周期及其核心概念,并深入源码分析。Spring Bean是Spring框架的核心,由容器管理其生命周期。从实例化到销毁,共经历十个阶段,包括属性赋值、接口回调、初始化及销毁等。通过剖析`BeanFactory`、`ApplicationContext`等关键接口与类,帮助你深入了解Spring Bean的管理机制。希望本文能助你更好地掌握Spring Bean生命周期。
72 1
|
1月前
|
Java Spring
获取spring工厂中bean对象的两种方式
获取spring工厂中bean对象的两种方式
37 1
|
1月前
|
XML Java 应用服务中间件
【Spring】运行Spring Boot项目,请求响应流程分析以及404和500报错
【Spring】运行Spring Boot项目,请求响应流程分析以及404和500报错
167 2
|
1月前
|
Java 开发者 Spring
Spring bean的生命周期详解!
本文详细介绍了Spring框架中的核心概念——Spring Bean的生命周期,包括实例化、属性赋值、接口回调、初始化、使用及销毁等10个阶段,并深入剖析了相关源码,如`BeanFactory`、`DefaultListableBeanFactory`和`BeanPostProcessor`等关键类与接口。通过理解这些核心组件,读者可以更好地掌握Spring Bean的管理和控制机制。
84 1
|
2月前
|
负载均衡 Java 网络架构
实现微服务网关:Zuul与Spring Cloud Gateway的比较分析
实现微服务网关:Zuul与Spring Cloud Gateway的比较分析
102 5