当前位置: 首页 > java >正文

Spring中bean的生命周期(笔记)

bean的生命周期,按照最重要五步

第一步:实例化bean,调用无参构造方法(通过BeanDefinition利用反射实例化Bean对象(无参数构造方法) 并通过推断构造方法...并放入三级缓存中..
第二步:给bean属性赋值(调用set方法)(依赖注入 循环依赖问题(三级缓存))
第三步:初始化bean(会调用bean的init方法,注意,这个init需要自己去配自己写)
第四步:使用bean
第五步:销毁bean (需要调用destroy方法,需要自己写,自己配)
 


public class User {private String username;public User() {System.out.println("第一步,无参构造执行");}public void setUserName(String username) {System.out.println("第二步,给构造方法中的属性赋值");this.username = username;}public void initBean(){System.out.println("第三步初始化Bean");}
-------------------------public void  destroy(){System.out.println("第五步销毁Bean");}
}
<!--    手动指定初始化方法和销毁方法--><bean id="userBean" class="com.bili.spirng.domain.User" init-method="initBean" destroy-method="destroy">在初始化之前执行,user方法都被调用了<property name="userName" value="cccccccc"></property></bean>
</beans>

七步: 在初始化的前和后

/**
 * 第一步:实例化bean,调用无参构造方法
 * 第二步:给bean属性赋值(调用set方法)
在这里会执行bean后处理器的Before方法

该接口的before和after方法里可以完成对bean属性的读取 修改等..
 * 第三步:初始化bean(会调用bean的init方法,注意,这个init需要自己去配自己写)
在这里会执行Bean后处理器的after方法

一般AOP是在这里实现的==>AbstractAutoProxyCreator()..
 * --------------到这步就基本完成了-------------------
 * 第四步:使用bean
 * 第五步:销毁bean (需要调用destroy方法,需要自己写,自己配)
 *
 */

<!--    强调这个处理器:将作用于整个当前配置文件的bean对象,会自己去调用,先自己写类--><bean id="LogBeanPostProcessor" class="com.bili.spirng.domain.LogBeanPostProcessor"></bean>
</beans>
/*** 日志后处理器* 这两个参数,第一个传入刚创建出来的bean对象,第二个参数传入bean的名称(会自己去传)* 但是你写好后还不行,还是得配进去让Spring管理器起来 注意一旦bean创建起来就会去自动执行,
比方说我们不需要再去user这个对象里面自己去配置before 以及 after 去调用*/
public class LogBeanPostProcessor implements BeanPostProcessor {@Overridepublic Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {System.out.println("执行bean:"+beanName+"后处理器的before");return BeanPostProcessor.super.postProcessBeforeInitialization(bean, beanName);}@Overridepublic Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {System.out.println("执行bean:"+beanName+"后处理器的after");return BeanPostProcessor.super.postProcessAfterInitialization(bean, beanName);}
}

十步:都是在检查是否对象实现了特定的接口,那么如果实现了,Spring会调用这个接口的方法
1.在bean后处理器before方法之前会检查是否实现aware接口
2.在bean后处理器before方法之后, 检查Bean是否实现了InitializingBean接口
3.使用bean之后,或者销毁bean前

/**
 * 第一步:实例化bean,调用无参构造方法
 * 第二步:给bean属性赋值(调用set方法)

(新增)before方法之前:

 这时候新实现了这三个之一的接口aware。每个对象都会去判断,如果说实现了,那我Spring就调用setBeanClassLoader,setBeanFactory或者setBeanName之一(这里我都写了)

作用:可以用来获取信息..

看你继承了哪个,重点并自己传入参数,假如说以后有个需求让你使用bean的名字那么我就去继承BeanNameAware去调用setBeanName的方法,那么spring已看你实现了接口,那我就去调这个方法,把这个名称传进去,不用我们人为传,就是传数据让你使用罢了吧。

在这里会执行bean后处理器的Before方法、

新增:before之后:执行InitializingBean(可以完成一些属性的赋值操作)

 * 第三步:初始化bean(会调用bean的init方法,注意,这个init需要自己去配自己写)
在这里会执行Bean后处理器的after方法
 * --------------到这步就基本完成了-------------------
 * 第四步:使用bean


检查Bean是否实现了DispoableBean接口,并执行销毁方法


 * 第五步:销毁bean (需要调用destroy方法,需要自己写,自己配)
 *
 */
 * 以上都可以在某个点位执行你想要的结果


public class User implements BeanFactoryAware , BeanClassLoaderAware, InitializingBean , DisposableBean {private String username;public User() {System.out.println("第一步,无参构造执行");}public void setUserName(String username) {this.username = username;System.out.println("第二步,给构造方法中的属性赋值 username:"+username);}public void initBean(){System.out.println("第三步初始化Bean");}public void  destroyBean(){System.out.println("第五步销毁Bean");}@Overridepublic void setBeanFactory(BeanFactory beanFactory) throws BeansException {System.out.println("生产"+beanFactory);}@Overridepublic void setBeanClassLoader(ClassLoader classLoader) {System.out.println(classLoader);}@Overridepublic void afterPropertiesSet() throws Exception {System.out.println("init被调用了");}@Overridepublic void destroy() throws Exception {System.out.println("销毁前");}
}

补充:

- BeanPostProcessor后置处理器是对所有Bean的初始化前后进行拦截...
- Aware InitializingBean等都是针对某个Bean的初始化前后做的处理...

http://www.xdnf.cn/news/2853.html

相关文章:

  • LeetCode热题100--53.最大子数组和--中等
  • 最新的30个Android Kotlin面试题
  • Kafka的Rebalance机制可能引发什么问题?如何优化?怎么减少不必要的Rebalance
  • 第十六届蓝桥杯 2025 C/C++组 密密摆放
  • Vue 中的过渡效果与响应式数据:transition、transitiongroup、reactive 和 ref 详解
  • FastGPT部署的一些问题整理
  • 对 FormCalc 语言支持较好的 PDF 编辑软件综述
  • 短视频矩阵批量剪辑与场景剪辑功能 OEM 定制开发
  • C++——调用OpenCV和NVIDIA Video Codec SDK库实现使用GPU硬解码MP4视频文件
  • 【深度学习与大模型基础】第14章-分类任务与经典分类算法
  • 从 BERT 到 GPT:Encoder 的 “全局视野” 如何喂饱 Decoder 的 “逐词纠结”
  • 高定电视,一场关于生活方式的觉醒
  • 在spark里通过jps命令,看到的进程
  • 【C++11】包装器:function与bind
  • iVX 引领软件开发进入 “可视化逻辑时代”
  • vue+cesium线流动纹理
  • WPF TextBlock控件性能优化指南
  • 【RK3588嵌入式图形编程】-Cairo-快速了解Cairo图形库
  • MongoDB的图形化工具robo3t,navicat
  • k8s-Pod生命周期
  • Tomcat 部署配置指南
  • java使用websocket推送消息到页面
  • 为什么执行了删除语句后mysql内存无变化?
  • Vue 2 和 Vue 3 中 Vue 实例变量方法的功能差异对比,包含关键方法的详细说明和表格总结
  • 硅基计划 学习总结 拾
  • 软考-软件设计师中级备考 7、算法设计与分析
  • 如何理解promise 续二
  • C语言学习路线
  • 国内外都有哪些医药医学方面的指南检索数据库?
  • 模电——PN结