SpringBoot中操作Bean的生命周期的方法

引言

在 Spring Boot 应用中,管理和操作 Bean 的生命周期是一项关键的任务。这不仅涉及到如何创建和销毁 Bean,还包括如何在应用的生命周期中对 Bean 进行精细控制。Spring 框架提供了多种机制来管理 Bean 的生命周期,这些机制使得开发者可以根据具体的业务需求和场景来定制 Bean 的行为。从简单的注解到实现特定的接口,每种方法都有其适用的场景和优势。

在 Spring Boot 中,操作 Bean 生命周期的方法主要包括以下:

1. InitializingBean 和 DisposableBean 接口:

在某些环境或特定的约束下,如果您想避免使用 JSR-250

  • InitializingBean 接口提供了一个方法 afterPropertiesSet(),该方法在 Bean 属性设置之后调用。

  • DisposableBean 接口提供了一个方法 destroy(),该方法在 Bean 销毁之前调用。

import org.springframework.beans.factory.DisposableBean;
import org.springframework.beans.factory.InitializingBean;public class MyBean implements InitializingBean, DisposableBean {@Overridepublic void afterPropertiesSet() throws Exception {// 初始化代码System.out.println("Bean is initialized");}@Overridepublic void destroy() throws Exception {// 清理代码System.out.println("Bean is destroyed");}
}

2. @PostConstruct 和 @PreDestroy 注解:

这两个是案例1中相对应的注解方式

  • @PostConstruct 注解用于在依赖注入完成后执行初始化方法。

  • @PreDestroy 注解用于在 Bean 销毁之前执行清理方法。

import javax.annotation.PostConstruct;
import javax.annotation.PreDestroy;public class MyBean {@PostConstructpublic void init() {// 初始化代码System.out.println("Bean is initialized");}@PreDestroypublic void cleanup() {// 清理代码System.out.println("Bean is destroyed");}
}

3. Bean 定义的 initMethod 和 destroyMethod

第三种方式的初始化和销毁方法

  • 在 Bean 定义中,可以通过 initMethod 和 destroyMethod 属性指定初始化和销毁方法。

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;@Configuration
public class AppConfig {@Bean(initMethod = "init", destroyMethod = "cleanup")public MyBean myBean() {return new MyBean();}public static class MyBean {public void init() {// 初始化代码System.out.println("Bean is initialized");}public void cleanup() {// 清理代码System.out.println("Bean is destroyed");}}
}

4. 实现 BeanPostProcessor 接口:

  • BeanPostProcessor 接口提供了两个方法:postProcessBeforeInitialization 和 postProcessAfterInitialization,分别在 Bean 初始化之前和之后调用。

  • 这可以用于在 Bean 初始化的不同阶段执行自定义逻辑。

import org.springframework.beans.factory.config.BeanPostProcessor;
import org.springframework.stereotype.Component;@Component
public class MyBeanPostProcessor implements BeanPostProcessor {@Overridepublic Object postProcessBeforeInitialization(Object bean, String beanName) {// 在初始化之前执行的代码return bean;}@Overridepublic Object postProcessAfterInitialization(Object bean, String beanName) {// 在初始化之后执行的代码return bean;}
}

5. 实现 SmartLifecycle 接口:

图片

  • SmartLifecycle 是一个扩展的接口,用于更复杂的生命周期管理,特别是在有多个 Bean 依赖关系的场景中。

  • 它提供了启动和停止控制,以及对应的回调方法。

import org.springframework.context.SmartLifecycle;
import org.springframework.stereotype.Component;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.util.concurrent.atomic.AtomicBoolean;@Component
public class MySmartLifecycleBean implements SmartLifecycle {private static final Logger logger = LoggerFactory.getLogger(MySmartLifecycleBean.class);private final AtomicBoolean isRunning = new AtomicBoolean(false);@Overridepublic void start() {// 启动逻辑if (isRunning.compareAndSet(false, true)) {// 实际的启动逻辑initializeResources();logger.info("Lifecycle bean started");}}@Overridepublic void stop() {// 停止逻辑if (isRunning.compareAndSet(true, false)) {// 实际的停止逻辑releaseResources();logger.info("Lifecycle bean stopped");}}@Overridepublic boolean isRunning() {return isRunning.get();}@Overridepublic int getPhase() {// 控制启动和停止的顺序return 0; // 默认阶段是 0,可以根据需要调整}private void initializeResources() {// 具体的资源初始化逻辑}private void releaseResources() {// 具体的资源释放逻辑}
}

6. 使用 ApplicationListener 或 @EventListener

  • 这些用于监听应用事件,如上下文刷新、上下文关闭等,可以在这些事件发生时执行特定逻辑。

  • ApplicationListener 是一个接口,而 @EventListener 是一个注解,两者都可以用于监听应用事件。

import org.springframework.context.ApplicationListener;
import org.springframework.context.event.ContextRefreshedEvent;
import org.springframework.stereotype.Component;@Component
public class MyApplicationListener implements ApplicationListener<ContextRefreshedEvent> {@Overridepublic void onApplicationEvent(ContextRefreshedEvent event) {// 在应用上下文刷新时执行的代码System.out.println("Application Context Refreshed");}
}// 或者使用 @EventListener
@Component
public class MyEventListener {@EventListenerpublic void handleContextRefresh(ContextRefreshedEvent event) {System.out.println("Handling context refreshed event.");}
}

7. 实现 ApplicationContextAware 和 BeanNameAware 接口:

  • 这些接口允许 Bean 在其生命周期内访问 ApplicationContext 和自身的 Bean 名称。

  • 通过实现这些接口,Bean 可以获得对 Spring 容器更深层次的访问和控制。

import org.springframework.beans.factory.BeanNameAware;
import org.springframework.context.ApplicationContextAware;
import org.springframework.context.ApplicationContext;
import org.springframework.stereotype.Component;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;@Component
public class MyAwareBean implements ApplicationContextAware, BeanNameAware {private static final Logger logger = LoggerFactory.getLogger(MyAwareBean.class);private ApplicationContext applicationContext;private String beanName;@Overridepublic void setApplicationContext(ApplicationContext applicationContext) {this.applicationContext = applicationContext;// 可以在这里执行与应用上下文相关的操作logger.info("ApplicationContext has been set for Bean: {}", beanName);}@Overridepublic void setBeanName(String name) {this.beanName = name;// 记录 Bean 名称logger.info("Bean name set to {}", name);}// 示例方法,展示如何使用 applicationContextpublic void performSomeAction() {try {// 示例逻辑,例如检索其他 Bean 或环境属性// String someProperty = applicationContext.getEnvironment().getProperty("some.property");// ... 执行操作} catch (Exception e) {logger.error("Error during performing some action", e);}}
}

8. 使用 FactoryBean

  • FactoryBean 是一种特殊的 Bean,用于生成其他 Bean。

  • 可以通过实现 FactoryBean 接口来控制 Bean 的实例化过程。

import org.springframework.beans.factory.FactoryBean;
import org.springframework.stereotype.Component;@Component
public class MyFactoryBean implements FactoryBean<MyCustomBean> {@Overridepublic MyCustomBean getObject() throws Exception {return new MyCustomBean();}@Overridepublic Class<?> getObjectType() {return MyCustomBean.class;}
}public class MyCustomBean {// 自定义 Bean 的逻辑
}

9. 使用 EnvironmentAware 和 ResourceLoaderAware 接口:

  • 这些接口允许 Bean 在其生命周期内访问 Spring 的 Environment 和资源加载器(ResourceLoader)。

  • 通过实现这些接口,Bean 可以获得对环境属性和资源的访问。

import org.springframework.context.EnvironmentAware;
import org.springframework.core.env.Environment;
import org.springframework.context.ResourceLoaderAware;
import org.springframework.core.io.ResourceLoader;
import org.springframework.stereotype.Component;@Component
public class MyEnvironmentAwareBean implements EnvironmentAware, ResourceLoaderAware {private Environment environment;private ResourceLoader resourceLoader;@Overridepublic void setEnvironment(Environment environment) {this.environment = environment;}@Overridepublic void setResourceLoader(ResourceLoader resourceLoader) {this.resourceLoader = resourceLoader;}
}

10. 实现 BeanFactoryAware 接口:

  • 通过实现 BeanFactoryAware 接口,Bean 可以访问到 Spring 容器中的 BeanFactory,从而可以进行更复杂的依赖注入和管理,BeanFactoryAware 应该在需要动态访问或管理 Bean 时作为特殊用例来使用。

import org.springframework.beans.factory.BeanFactory;
import org.springframework.beans.factory.BeanFactoryAware;
import org.springframework.stereotype.Component;@Component
public class MyBeanFactoryAware implements BeanFactoryAware {private BeanFactory beanFactory;@Overridepublic void setBeanFactory(BeanFactory beanFactory) {this.beanFactory = beanFactory;}
}

11. 使用 @Profile 注解:

  • @Profile 注解允许根据不同的环境配置(如开发、测试、生产)来激活或禁用特定的 Bean。

  • 这对于控制 Bean 在不同环境下的创建和管理非常有用。

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Profile;@Configuration
public class MyConfiguration {@Bean@Profile("development")public MyBean devMyBean() {return new MyBean();}@Bean@Profile("production")public MyBean prodMyBean() {return new MyBean();}public static class MyBean {// Bean 实现}
}

12. 使用 @Lazy 注解:

  • @Lazy 注解用于延迟 Bean 的初始化直到它被首次使用。

  • 这对于优化启动时间和减少内存占用非常有用,特别是对于那些不是立即需要的 Bean。

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Lazy;@Configuration
public class MyConfiguration {@Bean@Lazypublic MyBean myLazyBean() {return new MyBean();}public static class MyBean {// Bean 实现}
}

13. 使用 @DependsOn 注解:

  • @DependsOn 注解用于声明 Bean 的依赖关系,确保一个 Bean 在另一个 Bean 之后被初始化。

  • 这在管理 Bean 之间的依赖和初始化顺序时非常有用。

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.DependsOn;@Configuration
public class MyConfiguration {@Bean@DependsOn("anotherBean")public MyBean myBean() {return new MyBean();}@Beanpublic AnotherBean anotherBean() {return new AnotherBean();}public static class MyBean {// Bean 实现}public static class AnotherBean {// 另一个 Bean 实现}
}

14. 使用 @Order 或 Ordered 接口:

  • 这些用于定义 Bean 初始化和销毁的顺序。

  • @Order 注解和 Ordered 接口可以帮助确保 Bean 按照特定的顺序被创建和销毁。

import org.springframework.core.Ordered;
import org.springframework.core.annotation.Order;
import org.springframework.stereotype.Component;@Order(Ordered.HIGHEST_PRECEDENCE)
@Component
public class MyHighPriorityBean {// 高优先级 Bean 实现
}@Component
public class MyDefaultPriorityBean {// 默认优先级 Bean 实现
}

15. 使用 @Conditional 注解:

  • @Conditional 注解用于基于特定条件创建 Bean。

  • 你可以创建自定义条件或使用 Spring 提供的条件,如操作系统类型、环境变量、配置属性等。

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Conditional;
import org.springframework.context.annotation.Configuration;
import org.springframework.core.env.Environment;@Configuration
public class MyConfiguration {@Bean@Conditional(MyCondition.class)public MyBean myConditionalBean() {return new MyBean();}public static class MyBean {// Bean 实现}public static class MyCondition implements Condition {@Overridepublic boolean matches(ConditionContext context, AnnotatedTypeMetadata metadata) {Environment env = context.getEnvironment();// 定义条件逻辑return env.containsProperty("my.custom.condition");}}
}

总结

Spring Boot 提供的这些方法使得开发者能够灵活地控制 Bean 的生命周期,从而满足不同的应用需求和场景。无论是简单的应用还是复杂的企业级系统,合理地利用这些机制可以有效地管理 Bean 的生命周期,提高应用的性能和可维护性。选择哪种方法取决于具体的需求、应用的复杂性以及开发团队的偏好。正确地使用这些工具和技术可以使 Spring Boot 应用更加健壮、灵活和高效。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/588229.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Error: TF_DENORMALIZED_QUATERNION: Ignoring transform forchild_frame_id

问题 运行程序出现&#xff1a; Error: TF_DENORMALIZED_QUATERNION: Ignoring transform for child_frame_id “odom” from authority “unknown_publisher” because of an invalid quaternion in the transform (0.0 0.0 0.0 0.707) 主要是四元数没有归一化 Eigen::Quatern…

【Kotlin】委托模式

1 委托模式简介 委托模式的类图结构如下。 对应的 Kotlin 代码如下。 fun main() {var baseImpl BaseImpl()var baseWrapper BaseWrapper(baseImpl)baseWrapper.myFun1() // 打印: BaseImpl, myFun1baseWrapper.myFun2() // 打印: BaseImpl, myFun2 }interface Base {fun my…

【御控物联】JavaScript JSON结构转换(18):数组To对象——多层属性重组

文章目录 一、JSON结构转换是什么&#xff1f;二、案例之《JSON数组 To JSON对象》三、代码实现四、在线转换工具五、技术资料 一、JSON结构转换是什么&#xff1f; JSON结构转换指的是将一个JSON对象或JSON数组按照一定规则进行重组、筛选、映射或转换&#xff0c;生成新的JS…

golang语言系列:Web框架+路由 之 Echo

云原生学习路线导航页&#xff08;持续更新中&#xff09; 本文是golang语言系列文章&#xff0c;本篇主要对 Echo 框架 的基本使用方法 进行学习 1.Echo是什么 Go 有众多Web框架&#xff0c;Echo 是其中的一个&#xff0c;官网介绍Echo有高性能、可扩展性、极简的特点。使用E…

Object类的方法-clone() \ finalize()

了解方法&#xff1a;clone() \ finalize()根据JDK源代码及Object类的API文档&#xff0c;Object类当中包含的方法有11个。这里我们主要关注其中的6个&#xff1a; clone() //Object类的clone()的使用 public class CloneTest { public static void main(String[] args) { …

关于ITIL认证您需要了解的一切

这是一篇关于从业人员、领导者和 ITSM 爱好者指南。ITIL4于2019 年发布。最新版本的 IT 服务管理&#xff08;ITSM&#xff09;最佳实践从传统的生命周期方法转变为服务价值体系模型&#xff0c;重点关注价值共创、向业务交付成果以及与其他最佳实践框架的融合。 新版本的框架…

【教程】Kotlin语言学习笔记(五)——Lambda表达式与条件控制

写在前面&#xff1a; 如果文章对你有帮助&#xff0c;记得点赞关注加收藏一波&#xff0c;利于以后需要的时候复习&#xff0c;多谢支持&#xff01; 【Kotlin语言学习】系列文章 第一章 《认识Kotlin》 第二章 《数据类型》 第三章 《数据容器》 第四章 《方法》 第五章 《L…

Redission--布隆过滤器解决缓存穿透问题

布隆过滤器在缓存穿透问题中的使用 布隆过滤器的核心是一个位数组 布隆过滤器的误判 使用Redission的布隆过滤器步骤 添加 Redission 依赖&#xff1a;首先需要将 Redission 添加到你的 Java 项目中&#xff0c;你可以通过 Maven 来添加 Redission 的依赖。 创建 Redissio…

GT收发器PHY层设计(2)GT_module模块设计

文章目录 前言一、设计框图二、例化IP核端口三、common_reset_i模块四、gt_usrclk_source模块五、IBUFDS_GTE2和gtwizard_0_common模块六、顶层模块gt_module总结 前言 根据官方的example design设计一个自定义协议的高速PHY设计 一、设计框图 设计思路及代码思路参考FPGA奇哥…

九州未来深度参与元宇宙标准会议周

近日&#xff0c;元宇宙标准化工作组成立大会暨第一次全体委员会会议在浙江青田成功举办。本次会议由元宇宙标准化工作组主办&#xff0c;中国电子技术标准化研究院、中共青田县委 青田县人民政府承办&#xff0c;涵盖了《元宇宙参考架构》国家标准编制会、《工业元宇宙参考架构…

AI音乐GPT时刻来临:Suno 快速入门手册!

✨✨ 欢迎大家来访Srlua的博文&#xff08;づ&#xffe3;3&#xffe3;&#xff09;づ╭❤&#xff5e;✨✨ &#x1f31f;&#x1f31f; 欢迎各位亲爱的读者&#xff0c;感谢你们抽出宝贵的时间来阅读我的文章。 我是Srlua小谢&#xff0c;在这里我会分享我的知识和经验。&am…

铸铁平台划线基准的基本操作——泊头市北重厂家

划线基准是指在铸铁平台上划定一条直线或直角线&#xff0c;用作测量和定位的参考线。下面是铸铁平台划线基准的操作步骤&#xff1a; 准备工具和材料&#xff1a;铸铁平台、直尺、铅笔或划线笔、胶带等。 清洁铸铁平台&#xff1a;使用清洁布或刷子清除铸铁平台表面的尘土和杂…