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

全注解下的SpringIoc 续2-bean的生命周期

spring中bean的生命周期

上一个小节梳理了一下Spring Boot的依赖注入的基本知识,今天来梳理一下spring中bean的生命周期。
下面,让我们一起看看bean在IOC容器中是怎么被创建和销毁的。
bean的生命周期大致分为四个部分:

定义
初始化
生存期
销毁

bean的定义和初始化

大致流程如下:

Created with Raphaël 2.3.0 开始 资源定位(通过@ComponentScan注解定义的扫描规则去寻找被@Component、@Controller、@Service......等注解标记的类) 资源解析、并且将定义的信息保存起来 定义发布:将bean的定义信息发布到ioc容器中 bean的初始化 结束

默认情况下,spring会在启动时完成对bean的定义、发布以及初始化,但是,有时候我们并不想让spring在启动时就完成bean的初始化,更想的是在我们用到它时,才去完成初始化的动作,最常见的就是循环依赖的场景了。
而解决这个问题就需要用到spring的延迟初始化的机制了。

spring的延迟初始化bean机制

可以使用2中方法使bean进行延迟初始化:

@ComponentScan注解的lazyInit属性

@ComponentScan(basePackages = {"com.zzm.iocbeanlifeperiod"},lazyInit = true)

设置该属性为true后,对应的扫描规则下的bean都会进行延迟初始化

@Lazy注解

该注解用于指定某一个依赖的bean进行延迟初始化,用法如下:

	@Autowired@Lazyprivate Animal cat;

bean的初始化生命周期的各个阶段

好了,现在我们正式看看bean的生命周期的各个阶段,如下图所示:
在这里插入图片描述
这个流程介绍了bean从初始化到销毁的过程。
注意:流程中的setApplicationContext方法有些特殊,即使你定义了 ApplicationContextAware 接口,但是有时候并不会调用,这要根据你的 IoC器来决定。 我们知道, Spring IoC 容器最低的要求是实 BeanFactory 接口,而不是ApplicationContext 接口, 对于那些没有实现 Application Cont xt 接口的容器 ,在生命周期对应的 ApplicationContextAware 定义的方法也是不会被调用的,只有实现了 Applic ationContext 接口的容器,才会在生命周期调用 ApplicationContextAware 定义的 setApplicationContext
方法。

接下来,让我们一起测试一下bean的生命周期。
首先创建一个类,同时让它实现流程中的这些接口,代码如下:

@Component
@Slf4j
public class BeanLifePeriod implements BeanNameAware, BeanFactoryAware, ApplicationContextAware, InitializingBean, DisposableBean {@Overridepublic void setBeanFactory(BeanFactory beanFactory) throws BeansException {log.warn("【{}】调用了BeanFactoryAware的setBeanFactory",this.getClass().getSimpleName());}@Overridepublic void setBeanName(String name) {log.warn("【{}】调用了BeanNameAware的setBeanName",this.getClass().getSimpleName());}@Overridepublic void destroy() throws Exception {log.warn("【{}】调用了DisposableBean的destroy",this.getClass().getSimpleName());}@PreDestroypublic void destroyMyself() {log.warn("【{}】调用了注解@PreDestroy定义的自定义销毁方法",this.getClass().getSimpleName());}@Overridepublic void afterPropertiesSet() throws Exception {log.warn("【{}】调用了InitializingBean的afterPropertiesSet",this.getClass().getSimpleName());}@Overridepublic void setApplicationContext(ApplicationContext applicationContext) throws BeansException {log.warn("【{}】调用了ApplicationContextAware的setApplicationContext",this.getClass().getSimpleName());}@PostConstructpublic void init(){log.warn("【{}】调用了注解@PostConstruct定义的自定义初始化方法",this.getClass().getSimpleName());}
}

因为BeanPostProcessor接口的方法是针对于所有的bean的,所以我们这里单独创建一个类来实现它:

@Component
@Slf4j
public class BeanPostProcessorExample implements BeanPostProcessor {@Overridepublic Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {log.warn("BeanPostProcessor调用postProcessBeforeInitialization方法,参数【{},{}】", bean.getClass().getSimpleName(), bean);return bean;}@Overridepublic Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {log.warn("BeanPostProcessor调用postProcessAfterInitialization方法,参数【{},{}】", bean.getClass().getSimpleName(), bean);return bean;}
}

接下来,创建ioc配置类和测试类

@ComponentScan(basePackages = {"com.zzm.iocbeanlifeperiod"})
@Configuration
@Slf4j
public class LifePeriodAppConfig {}public class IocBeanLifePeriod {public static void main(String[] args) throws InterruptedException {AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(LifePeriodAppConfig.class);context.close();}
}

运行main方法,可以看到如下日志:
在这里插入图片描述
上面的流程可用于自定义的bean,但是有时候bean不是我们自己定义的,而是第三方的bean,那怎么办呢?
在小节全注解下的IOC中我们介绍了@Bean注解,它可以用来向容器发布第三方的bean,所以通过它,我们也可以自定义初始化和销毁的方法,用法如下:

@ComponentScan(basePackages = {"com.zzm.iocbeanlifeperiod"})
@Configuration
@Slf4j
public class LifePeriodAppConfig {@Bean(value = "lifePeriodUser",initMethod = "init",destroyMethod = "destroy")public LifePeriodUser getUser(){return new LifePeriodUser().setId(3L).setUserName("Bean手动注入").setNote("Bean手动注入");}
}@Data
@Accessors(chain = true)
@Slf4j
public class LifePeriodUser {private Long id;private String userName;private String note;public void init(){log.warn("【{}】调用了自定义用户初始化方法",this.getClass().getSimpleName());}public void destroy(){log.warn("【{}】调用了自定义用户销毁方法",this.getClass().getSimpleName());}
}

感兴趣的小伙伴可以自己尝试下,今天就到这里了。。。。。

http://www.lryc.cn/news/62101.html

相关文章:

  • 【VQ-VAE代码实战】Neural Discrete Representation Learning
  • gpt3.5和gpt4区别-gpt3.5和gpt4
  • java获取当前系统时间
  • pbootcms自动配图出图插件
  • 手动测试台架搭建,让你的车载测试更轻松
  • 分组双轴图:揭示数据中的关联性和趋势变化
  • MATLAB函数封装1:生成QT可以调用的.dll动态链接库
  • 【算法题】2400. 恰好移动 k 步到达某一位置的方法数目
  • 探索【Stable-Diffusion WEBUI】的插件:骨骼姿态(OpenPose)
  • MySQL数据落盘原理(redo、undo、binlog、2PC、double write等。)
  • 智加科技+舍弗勒,首发量产正向开发的智能重卡冗余转向
  • C++类的模拟实现
  • 耐腐蚀高速电动针阀在半导体硅片清洗机化学药液流量控制中的应用
  • 助力工业物联网,工业大数据之ODS层及DWD层建表语法【七】
  • Windows环境下C++ 安装OpenSSL库 源码编译及使用(VS2019)
  • TensorFlow高阶API和低阶API
  • 强训之【参数解析和跳石板】
  • Redis队列Stream、Redis多线程详解(三)
  • MySQL统计函数count详解
  • 实验04:图像压缩(DP算法)
  • 4.19--面试系列之真题版本--redis出现大key怎么解决?Redis 大 Key 对持久化有什么影响?
  • 新手在家做自媒体要如何起步?
  • 易基因:禾本科植物群落的病毒组丰度/组成与人为管理/植物多样性变化的相关性 | 宏病毒组
  • 华为OD机试——对称美学(通过率只有8.51%???)
  • 【三十天精通Vue 3】第十六天 Vue 3 的虚拟 DOM 原理详解
  • Arduino ESP8266通过udp获取时间以及同步本地时间方法
  • c/c++:char*定义常量字符串,strcmp()函数,strcpy()函数,寻找指定字符,字符串去空格
  • 2023年6月DAMA-CDGA/CDGP数据治理认证考试可报名地区公布
  • UDS的0x19服务介绍
  • QinQ技术与Portal技术