• 为什么要了解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. mybatis-plus 错误 java.lang.NoClassDefFoundError

    错误 java.lang.NoClassDefFoundError: org/apache/velocity/context/Context 使用mybatis-plus自动生成文件的时候,报下面的错 ...

  2. ARST 第五周打卡

    Algorithm : 做一个 leetcode 的算法题 /////////////////////////////////////////////////////////////////// // ...

  3. WebMvcConfigurationSupport跨域和fastjson全局替换

    @Configuration public class WarnWebMvcConfigurationSupport extends WebMvcConfigurationSupport { /** ...

  4. Array Product CodeForces - 1042C (细节)

    #include <iostream> #include <sstream> #include <algorithm> #include <cstdio> ...

  5. 使用python操作kafka

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

  6. poj 3320 复习一下尺取法

    尺取法(two point)的思想不难,简单来说就是以下三步: 1.对r point在满足题意的情况下不断向右延伸 2.对l point前移一步 3.  回到1 two point 对连续区间的问题求 ...

  7. wpf menuitem 简约显示的 template样式

    <ControlTemplate x:Key="MenuItemControlTemplate1" TargetType="{x:Type MenuItem}&qu ...

  8. vue入门:(计算属性和侦听器)

    methods watch computed 一.methods-方法 在数据渲染是需要基于多个数据时第一种方法,可以采用methods属性中的方法计算返回值来实现,先来看示例: <div id ...

  9. 搭建vue.js 的npm脚手架

    1.在cmd中,找到nodeJs安装的路径下,运行 vue -V,查看当前vue版本,如下图所示,表明已经安装过了. 2.没有安装,进行安装.在cmd中,找到nodeJs安装的路径下,运命令行 npm ...

  10. Vue异步请求最佳实践

    一.当前存在的问题 目前项目前端请求后台数据的方式是这样的: 页面中method中dispatch到action action调用mutation,请求axios 请求到数据后存储到state中 页面 ...