【Spring】Bean 的作用域和生命周期

简介: 1. Bean 的作用域1.1 通过一个案例来看 Bean 作用域的问题1.2 作用域的定义1.3 Spring Bean 支持的作用域(未介绍完全)1.4 修改 Bean 的作用域1.5 Bean 执行流程2. Bean 的生命周期2.1 Bean 的生命周期分为以下 5 大部分2.1.1 实例化 Bean(为 Bean 分配内存空间)2.1.2 设置属性(Bean 注入和装配)2.1.3 Bean 初始化2.1.4 Bean 使用2.1.5 销毁 Bean2.2 Bean 的生命周期执行流程2.3 生命周期演示

1. Bean 的作用域

1.1 通过一个案例来看 Bean 作用域的问题

假设现有一个公用的 Bean,提供给两个用户 A 和 B 使用,但是在使用途中 A 用户在 B 用户不知情下修改了公共 Bean 的数据,导致用户 B 拿到的 Bean 不是预设的 Bean


公共 Bean

@Slf4j
@Component
@Data
public class User {
    public int uid;
    public String name;
    public User() {
        this.uid = 1024;
        this.name = "java";
        log.info("User 的构造方法被调用,user 对象被创建 {}", this);
    }
    @Override
    public String toString() {
        return "User{" +
                "uid=" + uid +
                ", name='" + name + '\'' +
                ", hashCode()=" + this.hashCode() +
                '}';
    }
}

A 用户使用时,对 Bean 进行了修改

@Slf4j
@Component
public class CommandLine1 implements CommandLineRunner {
    private final User user;
    @Autowired
    public CommandLine1(User user) {
        this.user = user;
    }
    @Override
    public void run(String... args) throws Exception {
        user.setName("hsq");
        log.info("CommandLine-1 下的 user 是 {}", user);
    }
}

B 用户拿到 Bean 时

@Slf4j
@Component
public class CommandLine2 implements CommandLineRunner {
    private final User user;
    @Autowired
    public CommandLine2(User user) {
        this.user = user;
    }
    @Override
    public void run(String... args) throws Exception {
        log.info("CommandLine-2 下的 user 是 {}", user);
    }
}

23.png

原因分析


操作以上问题的原因是因为 Bean 默认情况下是单例状态(singleton),也就是所有人的使用的都是同一个对象,单例可以很大程度上提高性能,所以在 Spring 中 Bean 的作用域默认也是 singleton 单例模式。


1.2 作用域的定义

限定程序中变量的可用范围叫做作用域,或者说在源代码中定义变量的某个区域就叫做作用域。


而 Bean 的作用域是指 Bean 在 Spring 整个框架中的某种行为模式,比如 singleton 单例作用域,就 表示 Bean 在整个 Spring 中只有一份,它是全局共享的,那么当其他人修改了这个值之后,那么另一个人读取到的就是被修改的值。


1.3 Spring Bean 支持的作用域(未介绍完全)

singleton


单例,Spring 中默认的作用域


prototype


每次从 Spring 容器中 get Bean 对象,都会触发一次创建过程,每个对象都是独立的对象


context.getBean(User.class) <=> new User()


request


以请求为单位,一次请求过程中,从开始到结尾,期间 context.getBean() 拿到的都是同一个 Bean ,但如果不同请求,则获取对象不同


session


以用户 Session 为单位。每个用户都有自己独立的 Bean,context.getBean() 根据不同的 session,得到不同的 Bean


1.4 修改 Bean 的作用域

使用 @Scope() 注解修饰 Bean(参数填上作用域名),即可修改 Bean 的作用域,不加注解默认为 singleton


1.5 Bean 执行流程

启动容器,加载配置文件(类加载路径下的 XML 文件)

根据配置完成 Bean 初始化,扫描与 application 文件同包下的 @Controller、@Service、@Component、@Repository 注解

注册 Bean 到容器中,如果 Bean 需要使用其他 Bean 作为属性,需提前注入

将 Bean 注入到需要的类中


2. Bean 的生命周期

所谓的生命周期指的是一个对象从诞生到销毁的整个生命过程,我们把这个过程就叫做一个对象的生命周期。


2.1 Bean 的生命周期分为以下 5 大部分

2.1.1 实例化 Bean(为 Bean 分配内存空间)

2.1.2 设置属性(Bean 注入和装配)

2.1.3 Bean 初始化

实现了各种 Aware 通知的方法,如 BeanNameAware、InitializingBean 等的接口方法

执行 BeanPostProcessor 初始化前置方法;

执行 @PostConstruct 初始化方法,依赖注入操作之后被执行;

执行自己指定的 init-method 方法(如果有指定的话);

执行 BeanPostProcessor 初始化后置方法。

2.1.4 Bean 使用

2.1.5 销毁 Bean

销毁容器的各种方法,如 @PreDestroy、DisposableBean 接口方法、destroy-method。


2.2 Bean 的生命周期执行流程


24.png

2.3 生命周期演示

@Slf4j
@Component
public class LifeOfBean implements ApplicationContextAware, BeanNameAware, BeanClassLoaderAware, ResourceLoaderAware, InitializingBean {
    @Autowired
    public LifeOfBean() {
        log.info("LifeOfBean 的构造方法");
    }
    @Autowired
    public void setName(@Value("${custom-user.name}") String name) {
        log.info("LifeOfBean 的 setName(name = {}) 方法", name);
    }
    // 一定发生属性被注入之后,bean 被使用之前
    @PostConstruct  // 指定 init-method
    public void initMethod() {
        log.info("LifeOfBean 的 init-method 方法");
    }
    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        log.info("LifeOfBean 的 setApplicationContext(), applicationContext = {}", applicationContext);
    }
    @Override
    public void setBeanName(String name) {
        log.info("LifeOfBean 的 setBeanName(), beanName = {}", name);
    }
    @Override
    public void setBeanClassLoader(ClassLoader classLoader) {
        log.info("LifeOfBean 的 setBeanClassLoader(), classLoader = {}", classLoader);
    }
    @Override
    public void setResourceLoader(ResourceLoader resourceLoader) {
        log.info("LifeOfBean 的 setResourceLoader(), resourceLoader = {}", resourceLoader);
    }
    // 一定发生属性被注入之后,bean 被使用之前
    @Override
    public void afterPropertiesSet() throws Exception {
        log.info("LifeOfBean 的 afterPropertiesSet()");
    }
    // 销毁 Bean
    @PreDestroy
    public void preDestroy() {
        log.info("执行:preDestroy()");
    }
}
@Slf4j
@SpringBootApplication
public class BeanApplication {
    public static void main(String[] args) {
        ConfigurableApplicationContext context = SpringApplication.run(BeanApplication.class, args);
        log.info("Spring Application 启动结束");
        LifeOfBean bean = context.getBean(LifeOfBean.class);
        log.info("拿到手的 bean: {}", bean);
    }
}

25.png



目录
相关文章
|
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的一些子标签及注解的使用)
|
2月前
|
缓存 安全 Java
Spring框架中Bean是如何加载的?从底层源码入手,详细解读Bean的创建流程
从底层源码入手,通过代码示例,追踪AnnotationConfigApplicationContext加载配置类、启动Spring容器的整个流程,并对IOC、BeanDefinition、PostProcesser等相关概念进行解释
200 24
Spring框架中Bean是如何加载的?从底层源码入手,详细解读Bean的创建流程
|
28天前
|
Java 测试技术 Windows
咦!Spring容器里为什么没有我需要的Bean?
【10月更文挑战第11天】项目经理给小菜分配了一个紧急需求,小菜迅速搭建了一个SpringBoot项目并完成了开发。然而,启动测试时发现接口404,原因是控制器包不在默认扫描路径下。通过配置`@ComponentScan`的`basePackages`字段,解决了问题。总结:`@SpringBootApplication`默认只扫描当前包下的组件,需要扫描其他包时需配置`@ComponentScan`。
|
2月前
|
XML Java 数据格式
Spring IOC—基于XML配置Bean的更多内容和细节(通俗易懂)
Spring 第二节内容补充 关于Bean配置的更多内容和细节 万字详解!
210 18
Spring IOC—基于XML配置Bean的更多内容和细节(通俗易懂)
|
2月前
|
XML Java 数据格式
spring复习02,xml配置管理bean
详细讲解了Spring框架中基于XML配置文件管理bean的各种方式,包括获取bean、依赖注入、特殊值处理、属性赋值、集合类型处理、p命名空间、bean作用域及生命周期和自动装配。
spring复习02,xml配置管理bean
|
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月前
|
Java 开发者 Spring
Spring bean的生命周期详解!
本文详细介绍了Spring框架中的核心概念——Spring Bean的生命周期,包括实例化、属性赋值、接口回调、初始化、使用及销毁等10个阶段,并深入剖析了相关源码,如`BeanFactory`、`DefaultListableBeanFactory`和`BeanPostProcessor`等关键类与接口。通过理解这些核心组件,读者可以更好地掌握Spring Bean的管理和控制机制。
84 1
|
2月前
|
XML Java 数据格式
spring复习03,注解配置管理bean
Spring框架中使用注解配置管理bean的方法,包括常用注解的标识组件、扫描组件、基于注解的自动装配以及使用注解后的注意事项,并提供了一个基于注解自动装配的完整示例。
spring复习03,注解配置管理bean