Bean的生命周期

本文详细阐述了Spring Bean的生命周期,从实例化、属性设置、初始化到销毁的全过程。通过一个具体的例子展示了如何在各个阶段调用不同的方法,包括 Aware 接口、InitMethod、DestroyMethod、BeanPostProcessor等。同时,还演示了如何使用注解和自定义方法来控制Bean的行为。

描述

Bean 的生命周期,就是一个 Bean 从创建到销毁,所经历的各种方法调用。 简单的来说,一个Bean的生命周期分为四个阶段:
实例化(Instantiation)、 属性设置(populate)、初始化(Initialization)、销毁(Destruction)

在这里插入图片描述

实例化

程序启动后,Spring把注解或者配置文件定义好的Bean对象转换成一个BeanDefination对象,然后完成整个BeanDefination的解析和加载的过程。Spring获取到这些完整的对象之后,会对整个BeanDefination进行实例化操作,实例化是通过反射的方式创建对象。

属性设置

实例化后的对象被封装在BeanWrapper对象中,并且此时对象仍然是一个原生的状态,并没有进行依赖注入。Spring根据BeanDefinition中的信息进行依赖注入, populateBean方法来完成属性的注入。

初始化

1、调用Aware接口相关的方法:invokeAwareMethod(完成beanName, beanClassLoader, beanFactory对象的属性设置)
2、调用beanPostProcessor中的前置处理方法(applyBeanPostProcessorsBeforeInitialization)
3、调用InitMethod方法:invokeInitMethod(),判断是否实现了initializingBean接口,如果有,调用afterPropertiesSet方法,没有就不调用
4、调用BeanPostProcessor后置处理方法(applyBeanPostProcessorsAfterInitialization),Spring 的Aop就是在此处实现的

销毁

判断是否实现了DisposableBean接口,调用destoryMethod方法

代码验证

1、项目结构

在这里插入图片描述

2、bean的配置类

package com.tk.bean.config;
	
import com.tk.bean.entity.Book;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

/**
 * @author taoke
 * @desc 实体类配置
 * @date 2022/3/23
 */
@Configuration
public class BookConfig {

    @Bean(initMethod = "myPostConstruct", destroyMethod = "myPreDestroy")
    public Book getBook() {
        Book book = new Book();
        book.setBookName("水浒传");
        return book;
    }
}

3、实体类Book

package com.tk.bean.entity;
	
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.*;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;

import javax.annotation.PostConstruct;
import javax.annotation.PreDestroy;

/**
 * @author taoke
 * @desc 书
 * @date 2022/3/21
 */
public class Book implements BeanNameAware, BeanFactoryAware, ApplicationContextAware, InitializingBean, DisposableBean {

    private String bookName;

    public String getBookName() {
        return bookName;
    }

    public Book() {
        System.out.println("构造方法执行了。。。。 ");
    }

    public void setBookName(String bookName) {
        this.bookName = bookName;
        System.out.println("设置属性执行了,书名是: " + getBookName());
    }

    @Override
    public void setBeanName(String name) {
        System.out.println("setBeanName 方法执行了。。。");
    }

    @Override
    public void setBeanFactory(BeanFactory beanFactory) throws BeansException {
        System.out.println("setBeanFactory 方法执行了。。。");
    }

    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        System.out.println("setApplicationContext 方法执行了。。。");
    }

    //MyBeanPostProcessor.postProcessBeforeInitialization

    @PostConstruct
    public void springPostConstruct() {
        System.out.println("@PostConstruct 执行了。。。");
    }

    @Override
    public void afterPropertiesSet() {
        System.out.println("afterPropertiesSet 执行了。。。");
    }

    public void myPostConstruct() {
        System.out.println("自定义方法 myPostConstruct 执行了。。。");
    }

    //MyBeanPostProcessor.postProcessAfterInitialization
    //Book name = thing king in java

    @PreDestroy
    public void springPreDestroy() {
        System.out.println("@PreDestroy 方法执行了。。。");
    }

    @Override
    public void destroy() {
        System.out.println("destroy 方法执行了。。。");
    }

    public void myPreDestroy() {
        System.out.println("自定义方法 myPreDestroy 执行了。。。 ");
    }

}

4、BeanPostProcessor

package com.tk.bean.processor;
	
import com.tk.bean.entity.Book;
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.config.BeanPostProcessor;
import org.springframework.stereotype.Component;

/**
 * @author taoke
 * @desc 增强处理
 * @date 2022/3/21
 */
@Component
public class MyBeanPostProcessor implements BeanPostProcessor {

    @Override
    public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
        if (bean instanceof Book) {
            System.out.println("MyBeanPostProcessor 前置方法 postProcessBeforeInitialization 执行了。。。");
        }
        return bean;
    }

    @Override
    public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
        if (bean instanceof Book) {
            System.out.println("MyBeanPostProcessor 后置方法 postProcessAfterInitialization 执行了。。。");
        }
        return bean;
    }
}

5、启动类

package com.tk.bean;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;

/**
 * @author taoke
 * @desc 启动类
 * @date 2022/3/21
 */
@SpringBootApplication
public class BeanTestApplication {

    public static void main(String[] args) {
        SpringApplication.run(BeanTestApplication.class);
    }
}

6、运行结果

在这里插入图片描述

小结

main方法启动后,Spring读@Bean注解,将注解修饰的对象加载到IOC容器,IOC容器将其转化为BeanDefination对象,并进行实例化,实例化后封装为BeanWrapper对象。然后Spring调用populateBean方法对BeanDefination对象进行属性填充,再调用initializeBean方法完成一些Aware操作,然后调用beanPostProcessor中的前置处理方法,如果实现了initializingBean,则调用afterPropertiesSet方法,然后调用BeanPostProcessor后置处理方法,Aop就是在此处实现的。处理完成后,bean就处于一个就绪状态,等待被调用。销毁时判断是否实现了DisposableBean接口,调用destoryMethod方法。

Spring 框架中的 Bean 生命周期管理是其核心机制之一,涉及到从 Bean 的创建、初始化到销毁的全过程。Spring 容器通过一系列标准步骤和扩展点,确保 Bean 能够按照预期被创建、使用和销毁,同时为开发者提供了丰富的自定义机会。 ### 实例化阶段 在 Bean生命周期中,实例化是第一步,Spring 容器会根据配置元数据(如 XML 配置文件或注解)来创建 Bean 的实例。默认情况下,容器会调用无参构造函数来创建 Bean 对象。如果 Bean 的定义中指定了工厂方法,则会使用该方法进行实例化。这一阶段的扩展点包括 `InstantiationAwareBeanPostProcessor` 接口,它允许在实例化前后对 Bean 进行干预,例如返回一个代理对象[^1]。 ### 属性注入阶段 实例化完成后,Spring 容器会根据配置信息将属性值注入到 Bean 中。这包括对基本类型、集合、引用对象等的赋值。此外,Spring 还会处理自动装配(autowiring)逻辑,将依赖的其他 Bean 注入到当前 Bean 中。在此阶段,同样可以使用 `InstantiationAwareBeanPostProcessor` 来修改属性值或进行其他处理[^3]。 ### 初始化阶段 初始化阶段是 Bean 生命周期中最为复杂的部分,它包括以下几个关键步骤: 1. **设置 Bean 名称和上下文**:如果 Bean 实现了 `BeanNameAware` 接口,则容器会调用 `setBeanName()` 方法,将 Bean 的名称传入;如果实现了 `BeanFactoryAware` 接口,则会调用 `setBeanFactory()` 方法,将容器引用传入。 2. **前置处理**:`BeanPostProcessor` 接口的 `postProcessBeforeInitialization()` 方法会在初始化方法调用之前执行,允许对 Bean 进行修改。 3. **调用初始化方法**:可以通过 `@PostConstruct` 注解、`InitializingBean` 接口的 `afterPropertiesSet()` 方法,或在 XML 中配置 `init-method` 属性来指定初始化逻辑。 4. **后置处理**:`BeanPostProcessor` 的 `postProcessAfterInitialization()` 方法会在初始化方法执行之后调用,进一步修改 Bean 或进行其他操作[^1]。 ### 使用阶段 经过初始化后,Bean 已经准备好供应用程序使用。此时,Bean 可以作为依赖被注入到其他 Bean 中,也可以被直接调用。Spring 容器在此阶段不会对 Bean 进行任何干预,除非有 AOP 代理或其他增强逻辑介入。 ### 销毁阶段 当容器关闭或不再需要某个 Bean 时,Spring 会执行销毁逻辑以释放资源。销毁阶段的执行步骤如下: 1. **调用销毁方法**:可以通过 `@PreDestroy` 注解、实现 `DisposableBean` 接口的 `destroy()` 方法,或在 XML 中配置 `destroy-method` 属性来定义销毁逻辑。 2. **资源释放**:通常在此阶段关闭数据库连接、释放线程池等资源[^2]。 ### 扩展点与自定义 Spring 提供了多个扩展点,允许开发者在 Bean 生命周期的不同阶段插入自定义逻辑。常见的扩展接口包括: - `BeanPostProcessor`:用于在初始化前后修改 Bean。 - `BeanFactoryPostProcessor`:用于在容器加载 Bean 定义后修改配置元数据。 - `ApplicationListener`:用于监听容器事件,如上下文刷新、关闭等。 - `SmartLifecycle`:用于控制 Bean 的启动和关闭顺序。 通过合理利用这些扩展点,可以实现诸如 AOP 代理、日志记录、资源管理等高级功能,从而提升应用程序的灵活性和可维护性[^3]。 ### 示例代码:自定义初始化和销毁方法 ```java import javax.annotation.PostConstruct; import javax.annotation.PreDestroy; public class MyBean { @PostConstruct public void init() { System.out.println("Bean 初始化完成"); } public void doSomething() { System.out.println("Bean 正在执行任务"); } @PreDestroy public void destroy() { System.out.println("Bean 即将销毁"); } } ``` 在上述示例中,`@PostConstruct` 和 `@PreDestroy` 分别用于定义初始化和销毁方法,Spring 容器会在相应阶段调用这些方法。 ---
评论 3
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值