前言:在前几篇查漏补缺中,其实我们已经涉及到bean生命周期了,本篇内容进行详细分析。


首先看bean实例化过程:

分析:

bean实例化开始后

  • 注入对象属性后(前面IOC初始化十几篇文章)。
  • 检查激活Aware接口。
  • 执行BeanPostProcessor前置处理
  • 实例化bean,这里会检查InitializingBean和init-method方法。
  • 执行BeanPostProcessor后置处理
  • 注册销毁bean的回调方法(并不是必须)。
  • 使用bean。
  • 检查DisposableBean和destroy-method方法。

注意:Spring是不会主动去调用销毁方法,必须我们主动告知Spring容器。

示例

 public class BeanLifeCycle implements BeanNameAware, BeanClassLoaderAware, BeanFactoryAware, InitializingBean, DisposableBean {

     private String msg;

     public BeanLifeCycle() {
System.out.println("调用构造函数:BeanLifeCycle()");
} public String getMsg() {
return msg;
} public void setMsg(String msg) {
System.out.println("属性注入:setMsg");
this.msg = msg;
} @Override
public void setBeanName(String name) {
System.out.println("BeanNameAware#setBeanName 被调用");
} @Override
public void setBeanClassLoader(ClassLoader classLoader) {
System.out.println("BeanClassLoaderAware#setBeanClassLoader 被调用");
} @Override
public void setBeanFactory(BeanFactory beanFactory) throws BeansException {
System.out.println("BeanFactoryAware#setBeanFactory 被调用"); } @Override
public void afterPropertiesSet() throws Exception {
System.out.println("InitializingBean#afterPropertiesSet 被调用,并对msg属性进行修改");
this.msg = "InitializingBean#afterPropertiesSet方法对msg属性进行修改";
} @Override
public void destroy() throws Exception {
System.out.println("DisposableBean#destroy 被调用");
} public void initMethod() {
System.out.println("init-method 被调用,并对msg属性进行修改,调用顺序在InitializingBean之后");
this.msg = "init-method方法对msg属性进行修改";
} public void destroyMethod() {
System.out.println("destroy-method 被调用,调用顺序在DisposableBean之后");
}
}

上述为主测试类,另外为了测试BeanPostProcessor,我们还需增加如下测试类:

 public class BeanLifeCycleBeanPostProcessor implements BeanPostProcessor {

     @Override
public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
System.out.println("beanName=" + beanName + " 初始化之前调用:BeanPostProcessor#postProcessBeforeInitialization!!!!!");
return bean;
} @Override
public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
System.out.println("beanName=" + beanName + " 初始化之后调用:BeanPostProcessor#postProcessAfterInitialization!!!!!");
return bean;
}
}

具体配置如下:

 <bean id="beanLifeCycle" class="com.dev.basebean.lifecycle.BeanLifeCycle"
init-method="initMethod" destroy-method="destroyMethod" p:msg="属性注入"/>
<bean id="beanLifeCycleBeanPostProcessor" class="com.dev.basebean.lifecycle.BeanLifeCycleBeanPostProcessor"/>

测试代码:

     @Test
public void beanLifeCycleTest() {
ApplicationContext context = new ClassPathXmlApplicationContext("classpath*:com/dev/config/lifecycle/lifecycle.xml");
BeanLifeCycle beanLifeCycle = context.getBean(BeanLifeCycle.class);
System.out.println("msg=" + beanLifeCycle.getMsg());
((ClassPathXmlApplicationContext) context).close();
}

运行结果如下:

通过以上运行结果,我们可知Bean的生命周期,总结如下:

  1. Spring容器根据实例化策略对Bean进行实例化(调用其构造方法)。
  2. 实例化完成后,如果该bean设置了一些属性,则利用set方法进行属性设置。
  3. 如果该bean实现了BeanNameAware接口,则调用其setBeanName方法。
  4. 如果该bean实现了BeanClassLoaderAware接口,则调用其setBeanClassLoader方法。
  5. 如果该bean实现了BeanFactoryAware接口,则调用其setBeanFactory方法。
  6. 如果容器中注册了BeanPostProcessor,则会调用其postProcessBeforeInitialization方法,完成bean前置处理。
  7. 如果该bean实现了InitializingBean接口,则调用其afterPropertiesSet方法。
  8. 如果该bean配置了init-method方法,则调用其指定的方法。
  9. 初始化完成后,如果容器中注册了BeanPostProcessor,则会调用其postProcessAfterInitialization方法,完成bean后置处理。
  10. 对象初始化完成,就可以使用bean了(对其进行各种骚操作,上述示例中调用了getMsg方法)。
  11. 在容器关闭之前,如果该bean实现了DisposableBean接口,则调用其destroy方法。
  12. 在容器关闭之前,如果该bean配置了destroy-method方法,则调用其指定方法。

至此bean的生命周期结束,该bean完成了其伟大的一生。


by Shawn Chen,2019.05.05,下午。

【spring源码分析】IOC容器初始化——查漏补缺(四)的更多相关文章

  1. 【spring源码分析】IOC容器初始化——查漏补缺(一)

    前言:在[spring源码分析]IOC容器初始化(十一)中提到了初始化bean的三个步骤: 激活Aware方法. 后置处理器应用(before/after). 激活自定义的init方法. 这里我们就来 ...

  2. 【spring源码分析】IOC容器初始化——查漏补缺(五)

    前言:我们知道在Spring中经常使用配置文件的形式对进行属性的赋值,那配置文件的值是怎么赋值到属性上的呢,本文将对其进行分析. 首先了解一个类:PropertySourcesPlaceholderC ...

  3. 【spring源码分析】IOC容器初始化——查漏补缺(二)

    前言:在[spring源码分析]IOC容器初始化(八)中多次提到了前置处理与后置处理,本篇文章针对此问题进行分析.Spring对前置处理或后置处理主要通过BeanPostProcessor进行实现. ...

  4. SPRING源码分析:IOC容器

    在Spring中,最基本的IOC容器接口是BeanFactory - 这个接口为具体的IOC容器的实现作了最基本的功能规定 - 不管怎么着,作为IOC容器,这些接口你必须要满足应用程序的最基本要求: ...

  5. 【spring源码分析】IOC容器初始化——查漏补缺(三)

    前言:本文分析InitializingBean和init-method方法,其实该知识点在AbstractAutowireCapableBeanFactory#initializeBean方法中有所提 ...

  6. Spring源码解析-ioc容器的设计

    Spring源码解析-ioc容器的设计 1 IoC容器系列的设计:BeanFactory和ApplicatioContext 在Spring容器中,主要分为两个主要的容器系列,一个是实现BeanFac ...

  7. spring源码分析---IOC(1)

    我们都知道spring有2个最重要的概念,IOC(控制反转)和AOP(依赖注入).今天我就分享一下spring源码的IOC. IOC的定义:直观的来说,就是由spring来负责控制对象的生命周期和对象 ...

  8. spring 源码之 ioc 容器的初始化和注入简图

    IoC最核心就是两个过程:IoC容器初始化和IoC依赖注入,下面通过简单的图示来表述其中的关键过程:

  9. Spring源码阅读-IoC容器解析

    目录 Spring IoC容器 ApplicationContext设计解析 BeanFactory ListableBeanFactory HierarchicalBeanFactory Messa ...

随机推荐

  1. 认识:before和:after伪类

    有时候,我们需要大量的重复代码去实现一个非常简单的功能,这不仅浪费时间,而且效率低下,例如: <div class="aa">你好</div><div ...

  2. 逐步探究ObjC的Weak技术底层

    前言 之前的文章有说过 Atomic 原子操作的原理,其作为一个特殊的修饰前缀,影响了存取操作. 在属性修饰定义中,还有另一类修饰前缀,他们分别是 strong weak assign copy,这些 ...

  3. Oracle查看表之间的约束

    ----查看表约束 表格: user_constraints 查询外键约束条件 select ' select count(*) from '||TABLE_NAME||';'from user_co ...

  4. js 小细节(持续更新)

    1.在对数据进行操作时,一定要考虑数据里面每一个value值是否存在 $.each(data, function(i, item){ if(item == null || item.firstFram ...

  5. JLINK固件烧写

    最近在使用uVision V5.14.0.0 的时候,由于我使用的Jlink是盗版的,导致软件总是退出,然后再网上找到了解决办法. 下面介绍解决办法: 参考: http://www.9mcu.com/ ...

  6. keil 选项卡设置

    *1.optimization  : level2. *2. 2)硬件目标设置选项卡(Target),见图6所示. 图6 1:选择硬件目标设置选项卡 2:指定用于的晶振频率 3:在应用中可以选择实时操 ...

  7. Java 基础 面向对象: 接口(interface )概念 以及接口之练习3 -定义一个接口用来实现两个对象的比较并 判断instanceof是否为同类

    接口(interface )概念概念笔记 及测试代码: /** 面向对象: 接口(interface ) 是与类并行的一个概念: * 1.接口可以看成一个特殊的抽象类,它是常量与抽象方法的一个集合 * ...

  8. Vue-cli中的proxyTable解决开发环境的跨域问题

    https://blog.csdn.net/u012149969/article/details/80288126 https://vuejs-templates.github.io/webpack/ ...

  9. 大二小学期C#资产管理大作业小记

    说明 这个程序是我大二夏季学期(俗称小学期)用Visual Studio + C#写的<资产管理>大作业.这个项目非常简单,就是用C#写出一个UI界面,并连接数据库进行增删改查.这是我第一 ...

  10. python中的几种数据类型(一)

    一.整型(数字)      python2中有长整形long      python3 中全都是整型 int n = 56 print(n.bit_length()) # 0011 1000 # 12 ...