• 为什么要了解Spring中 bean的生命周期?

有时候我们需要自定义bean的创建过程,因此了解Spring中 bean的生命周期非常重要。

  • 二话不说先上图:

  • 在谈具体流程之前先看看Spring官方文档中在介绍原型作用域时的一段话:In contrast to the other scopes, Spring does not manage the complete lifecycle of a prototype bean. The container instantiates, configures, and otherwise assembles a prototype object and hands it to the client, with no further record of that prototype instance. Thus, although initialization lifecycle callback methods are called on all objects regardless of scope, in the case of prototypes, configured destruction lifecycle callbacks are not called. The client code must clean up prototype-scoped objects and release expensive resources that the prototype beans hold. To get the Spring container to release resources held by prototype-scoped beans, try using a custom bean post-processor, which holds a reference to beans that need to be cleaned up.

也就是说Spring并不会管理原型作用域对象的完整生命周期,对象在被Spring处理到上图中的“bean可以使用了”之后就脱离了spring监管,不会调用销毁方法(如果配置了),除非自定义BeanPostProcessor增加想要的销毁逻辑。

  • 接下来介绍上图中的每个步骤:
  1. Spring对bean进行实例化。
  2. Spring将值及bean的引用注入到当前bean对应的属性中。
  3. 如果bean实现了BeanNameAware接口,那么Spring将bean的ID传给setBeanName()方法。
  4. 如果bean实现了BeanFactoryAware接口,那么Spring将调用setBeanFactory()方法,同时传入BeanFactory容器实例。
  5. 如果bean实现了ApplicationContextAware接口,那么Spring将调用setApplicationContext()方法,同时传入bean所在应用上下文的引用。
  6. 如果bean实现了BeanPostProcessor接口,那么Spring将调用他们的postProcessBeforeInitialization()方法。
  7. 如果bean实现了InitializingBean接口,那么Spring将调用他们的afterPropertiesSet()方法。类似地,如果<bean>中配置了init-method属性(即指定了自定义的初始化方法,该方法可以抛异常但是不能有参数),则还会接着调用该方法。
  8. 如果bean实现了BeanPostProcessor接口,那么Spring将调用他们的postProcessAfterInitialization()方法。
  9. bean准备就绪,可以使用了,他们将一直驻留与应用上下文,直到应用上下文被销毁。
  10. 如果bean实现了DisposableBean接口,那么Spring将调用他的destroy()方法。类似地,如果<bean>中配置了destroy-method属性(即指定了自定义的销毁方法,和步骤7中的init-method方法一样,该方法可以抛异常但是不能有参数),则还会接着调用该方法。
 package com.spring.beans.cycle;

 import org.springframework.beans.BeansException;
import org.springframework.beans.factory.BeanFactory;
import org.springframework.beans.factory.BeanFactoryAware;
import org.springframework.beans.factory.BeanNameAware;
import org.springframework.beans.factory.DisposableBean;
import org.springframework.beans.factory.InitializingBean;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware; public class LolService
implements InitializingBean, DisposableBean, BeanNameAware, BeanFactoryAware, ApplicationContextAware { // 实现DisposableBean接口的方法
@Override
public void destroy() throws Exception {
System.out.println("执行DisposableBean接口的destroy方法");
} // 实现InitializingBean接口的方法
@Override
public void afterPropertiesSet() throws Exception {
System.out.println("执行InitializingBean接口的afterPropertiesSet方法");
} // 通过配置文件中<bean>的init-method属性指定的初始化方法
public void init() {
System.out.println("init...");
} // 通过配置文件中<bean>的destroy-method属性指定的销毁方法
public void destroy0() {
System.out.println("destroy0...");
} @Override
public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
System.out.println("执行ApplicationContextAware接口的setApplicationContext()方法: " + applicationContext);
} @Override
public void setBeanFactory(BeanFactory beanFactory) throws BeansException {
System.out.println("执行BeanFactoryAware接口的setBeanFactory()方法: " + beanFactory);
} @Override
public void setBeanName(String name) {
System.out.println("执行BeanNameAware接口的setBeanName()方法: " + name);
} }

示例bean:LolService

package com.spring.beans.cycle;

import org.springframework.beans.BeansException;
import org.springframework.beans.factory.config.BeanPostProcessor; public class MyBeanPostProcessor implements BeanPostProcessor { @Override
public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
System.out.println("postProcessBeforeInitialization: " + bean + ", " + beanName);
return bean;
} @Override
public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
System.out.println("postProcessAfterInitialization: " + bean + ", " + beanName);
return bean;
} }

BeanPostProcessor实现类

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd"> <bean id="lolService" class="com.spring.beans.cycle.LolService" init-method="init" destroy-method="destroy0"></bean> <bean class="com.spring.beans.cycle.MyBeanPostProcessor"></bean>
</beans>

配置文件

package com.spring.beans.cycle;

import org.springframework.context.support.ClassPathXmlApplicationContext;

public class Main {

    public static void main(String[] args) {
ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("beans-cycle.xml");
LolService service = (LolService) context.getBean("lolService");
context.close();
} }

测试类

Dec 23, 2019 6:38:02 PM org.springframework.context.support.ClassPathXmlApplicationContext prepareRefresh
INFO: Refreshing org.springframework.context.support.ClassPathXmlApplicationContext@29ca901e: startup date [Mon Dec 23 18:38:02 CST 2019]; root of context hierarchy
Dec 23, 2019 6:38:02 PM org.springframework.beans.factory.xml.XmlBeanDefinitionReader loadBeanDefinitions
INFO: Loading XML bean definitions from class path resource [beans-cycle.xml]
执行BeanNameAware接口的setBeanName()方法: lolService
执行BeanFactoryAware接口的setBeanFactory()方法: org.springframework.beans.factory.support.DefaultListableBeanFactory@7e0b85f9: defining beans [lolService,com.spring.beans.cycle.MyBeanPostProcessor#0]; root of factory hierarchy
执行ApplicationContextAware接口的setApplicationContext()方法: org.springframework.context.support.ClassPathXmlApplicationContext@29ca901e: startup date [Mon Dec 23 18:38:02 CST 2019]; root of context hierarchy
postProcessBeforeInitialization: com.spring.beans.cycle.LolService@77847718, lolService
执行InitializingBean接口的afterPropertiesSet方法
init...
postProcessAfterInitialization: com.spring.beans.cycle.LolService@77847718, lolService
Dec 23, 2019 6:38:03 PM org.springframework.context.support.ClassPathXmlApplicationContext doClose
INFO: Closing org.springframework.context.support.ClassPathXmlApplicationContext@29ca901e: startup date [Mon Dec 23 18:38:02 CST 2019]; root of context hierarchy
执行DisposableBean接口的destroy方法
destroy0...

控制台

Spring中 bean的生命周期的更多相关文章

  1. JAVA面试题:Spring中bean的生命周期

    Spring 中bean 的生命周期短暂吗? 在spring中,从BeanFactory或ApplicationContext取得的实例为Singleton,也就是预设为每一个Bean的别名只能维持一 ...

  2. 深入理解Spring中bean的生命周期

    [Spring中bean的生命周期] bean的生命周期 1.以ApplocationContext上下文单例模式装配bean为例,深入探讨bean的生命周期: (1).生命周期图: (2).具体事例 ...

  3. Spring中Bean的生命周期及其扩展点

    原创作品,可以转载,但是请标注出处地址http://www.cnblogs.com/V1haoge/p/6106456.html Spring中Bean的管理是其最基本的功能,根据下面的图来了解Spr ...

  4. 简:Spring中Bean的生命周期及代码示例

    (重要:spring bean的生命周期. spring的bean周期,装配.看过spring 源码吗?(把容器启动过程说了一遍,xml解析,bean装载,bean缓存等)) 完整的生命周期概述(牢记 ...

  5. 通过BeanPostProcessor理解Spring中Bean的生命周期

    通过BeanPostProcessor理解Spring中Bean的生命周期及AOP原理 Spring源码解析(十一)Spring扩展接口InstantiationAwareBeanPostProces ...

  6. 一分钟掌握Spring中bean的生命周期!

    Spring 中bean 的生命周期短暂吗? 在spring中,从BeanFactory或ApplicationContext取得的实例为Singleton,也就是预设为每一个Bean 的别名只能维持 ...

  7. Spring中bean的生命周期!

    Spring 中bean 的生命周期短暂吗? 在spring中,从BeanFactory或ApplicationContext取得的实例为Singleton,也就是预设为每一个Bean的别名只能维持一 ...

  8. Spring官网阅读(十)Spring中Bean的生命周期(下)

    文章目录 生命周期概念补充 实例化 createBean流程分析 doCreateBean流程分析 第一步:factoryBeanInstanceCache什么时候不为空? 第二步:创建对象(crea ...

  9. 如果你每次面试前都要去背一篇Spring中Bean的生命周期,请看完这篇文章

    前言 当你准备去复习Spring中Bean的生命周期的时候,这个时候你开始上网找资料,很大概率会看到下面这张图: 先不论这张图上是否全面,但是就说这张图吧,你是不是背了又忘,忘了又背? 究其原因在于, ...

随机推荐

  1. linux系统设置允许密码登录

    编辑  /etc/ssh/sshd_config 文件 将PasswordAuthentication 的值改为 yes 然后重启ssh 服务 进入到  /etc/init.d 文件夹内 执行 ./s ...

  2. 同一台服务器请求easyswoole的一个websocket接口报错

    求助大神啊!file_get_contents报这个错:failed to open stream: Connection timed out换成curl又报这个错:couldn't connect ...

  3. 使用python操作kafka

    使用python操作kafka目前比较常用的库是kafka-python库 安装kafka-python pip3 install kafka-python 生产者 producer_test.py ...

  4. C#中判断文件夹或文件是否存在的方法

    一.根据虚拟路径获取文件物理路径: string savePath = Server.MapPath("~/Uploads/RemoteDatum/"); 二.判断文件夹是否存在 ...

  5. Func<>委托、扩展方法、yield、linq ForEach综合运用

    1.先定义一个Model类    public class P1    {        public string name { get; set; }        public int age ...

  6. [转载]flex中的正则表达式

    原文:https://blog.csdn.net/hczhiyue/article/details/20483209 (1)单字符匹配* ‘x’ 匹配字符 x.* ‘.’ 匹配任意一个字符(字节),除 ...

  7. Redis之淘汰策略

    Redis 内存数据集大小上升到一定大小的时候,就会进行数据淘汰策略. Redis 提供了 6 种数据淘汰策略: 1. volatile-lru:从已设置过期时间的数据集中挑选最近最少使用的数据淘汰. ...

  8. 正着打星星(js)

    //让用户输入行数,使用for循环嵌套打出正着的星星来,行数等于用户输入的数字 //例如:用户输入6 // * // *** // ***** // ******* // ********* // * ...

  9. C++虚函数和纯虚函数的用法和区别

    C++虚函数与纯虚函数用法与区别(转)   1. 虚函数和纯虚函数可以定义在同一个类(class)中,含有纯虚函数的类被称为抽象类(abstract class),而只含有虚函数的类(class)不能 ...

  10. beego注解路由不刷新(不生效)

    本文主要说明本人在使用beego的注解路由时不生效问题 背景: 1.按照官网进行注解路由配置,第一次设置路由,完全正确,注解路由可用. 2.修改路由注释后,发现swagger页面并未有对应的更新 3. ...