2022-08-30
原文作者:键盘林

bean的生命周期:

bean创建---初始化---销毁的过程

容器管理bean的生命周期;

我们可以自定义初始化和销毁方法;容器在bean进行当前生命周期的时候来调用我们自定义的初始化和销毁方法

1)指定初始化和销毁方法:注解参数中指定init-method和destroy-method=“”

创建Car类

202203062030426871.png

创建配置类@Configuration

加入car的bean方法

202203062030447352.png

创建IOCTest_LifeCycle类

202203062030461963.png

202203062030476394.png

运行:

202203062030488855.png

总结:

构造(对象创建)

单实例:

懒汉式:在被调用的时候创建,单例

饿汉式:在容器启动的时候创建

多实例:在每次获取的时候创建对象,而且是多例

既然我们之前在car类中创建了销毁方法、初始化方法,那我们如果在bean的生命周期中去调用它的呢?

修改MainConfigOfLifeCycle类中car()方法@Bean的配置

202203062030504826.png

再次运行:

虽然程序结束了,但是bean在程序结束的时候并没有被销毁

202203062030522247.png

在程序结束之后一定要调用close方法

202203062030585998.png

运行:

202203062031000849.png

总结:

销毁:

单实例:容器关闭的时候

多实例:容器不会管理这个bean;容器不会调用销毁方法,需要手动去销毁

修改bean的@Scope

2022030620310106510.png

修改IOCTest_LifeCycle:

2022030620310335111.png

运行:

2022030620310517912.png

我们可以看到它不进行 销毁操作

以上方式使用的是bean里面的参数调用,现在使用另外一种方式

2.类基础上:通过Bean实现InitializingBean(定义初始化逻辑)、DisposableBean(实现销毁逻辑)

创建Cat类继承InitializingBean,DisposableBean

2022030620310646213.png

2022030620310820614.png

配置组件注解

2022030620310996315.png

在配置文件中直接添加扫描注解

2022030620311198016.png

2022030620311327517.png

IOCTest_LifeCycle类

2022030620311484418.png

运行结果:

2022030620311629319.png

2022030620311749720.png

3)可以使用JSR250

第一个注解@PostConstruct:在bean创建完成并且属性赋值完成;来执行初始化

2022030620311838621.png

第二个注解@PreDestroy在容器销毁bean之前通知我们进行清理工作

2022030620312033522.png

创建Dog类

2022030620312209323.png

2022030620312381424.png

添加到容器中:

2022030620312555425.png

运行结果:

2022030620312698826.png

2022030620312812427.png

4)BeanPostProcessor:bean的后置处理器;在bean初始化前后进行一些处理工作;

2022030620312899528.png

它的两个方法:

postProcessBeforeInitialization:在初始化之前工作

postProcessAfterInitialization:在初始化之后工作

创建MyBeanPostProcessor类实现BeanPostProcessor接口,实现它的两个方法

2022030620313116329.png

打印传入的参数

2022030620313290430.png

返回bean(也可以返回经过处理的bean)

2022030620313429631.png

将后置处理器加入到组件中

2022030620313590132.png

打印测试:

2022030620313749533.png

该后置处理器是对所有自定义注册的bean都进行操作,一定要记住spring自身默认的bean是不经过我们自定义的后置处理器;

而且对于里面的操作是:先执行postProcessAfterInitialization(初始化之前)在执行postConstruct(初始化)最后执行postProcessAfterInitialization(初始化之后)

通过debug发现它的执行流程为:

beanPostProcessor原理

populateBean(beanName,mbd,instanceWrapper);给bean进行属性赋值

poplateBean(beanNae,mbd,instanceWrapper);给bean进行属性赋值

initializeBean

{

applyBeanPostProcessorsBeforeInitializetion();

invokeInitMethods();执行自定义初始化

applyBeanPostProcessorsAfterInitialization()

}

beanPostProcessor底层源码实现

要观察beanPostProcessor的整个实现这周期,直接看它的子类实现(ApplicationContextAware只是它的其中一个实现)

在Dog类中使它继承ApplicationContextAware接口,实现它的setApplicationContext方法

2022030620313958334.png

2022030620314107735.png

查看ApplicationContextAwareProcessor源码,我们可以看到它实现的是beanPostProcessor

2022030620314250336.png

查看它的方法

2022030620314524037.png

经过一系列判断然后调用invokAwareInteraces()

2022030620315070338.png

判断bean的类型并设置

2022030620315237539.png

总结:spring底层对Bean的使用:

bean赋值、注入其他组件、@Autowired、生命周期注解功能、@Async等等。

阅读全文