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

news2024/11/27 4:03:46

引言

在 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 {

    @Override
    public void afterPropertiesSet() throws Exception {
        // 初始化代码
        System.out.println("Bean is initialized");
    }

    @Override
    public 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 {

    @PostConstruct
    public void init() {
        // 初始化代码
        System.out.println("Bean is initialized");
    }

    @PreDestroy
    public 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 {

    @Override
    public Object postProcessBeforeInitialization(Object bean, String beanName) {
        // 在初始化之前执行的代码
        return bean;
    }

    @Override
    public 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);

    @Override
    public void start() {
        // 启动逻辑
        if (isRunning.compareAndSet(false, true)) {
            // 实际的启动逻辑
            initializeResources();
            logger.info("Lifecycle bean started");
        }
    }

    @Override
    public void stop() {
        // 停止逻辑
        if (isRunning.compareAndSet(true, false)) {
            // 实际的停止逻辑
            releaseResources();
            logger.info("Lifecycle bean stopped");
        }
    }

    @Override
    public boolean isRunning() {
        return isRunning.get();
    }

    @Override
    public 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> {

    @Override
    public void onApplicationEvent(ContextRefreshedEvent event) {
        // 在应用上下文刷新时执行的代码
        System.out.println("Application Context Refreshed");
    }
}

// 或者使用 @EventListener
@Component
public class MyEventListener {

    @EventListener
    public 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;

    @Override
    public void setApplicationContext(ApplicationContext applicationContext) {
        this.applicationContext = applicationContext;
        // 可以在这里执行与应用上下文相关的操作
        logger.info("ApplicationContext has been set for Bean: {}", beanName);
    }

    @Override
    public void setBeanName(String name) {
        this.beanName = name;
        // 记录 Bean 名称
        logger.info("Bean name set to {}", name);
    }

    // 示例方法,展示如何使用 applicationContext
    public 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> {

    @Override
    public MyCustomBean getObject() throws Exception {
        return new MyCustomBean();
    }

    @Override
    public 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;

    @Override
    public void setEnvironment(Environment environment) {
        this.environment = environment;
    }

    @Override
    public 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;

    @Override
    public 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
    @Lazy
    public 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();
    }

    @Bean
    public 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 {

        @Override
        public boolean matches(ConditionContext context, AnnotatedTypeMetadata metadata) {
            Environment env = context.getEnvironment();
            // 定义条件逻辑
            return env.containsProperty("my.custom.condition");
        }
    }
}

总结

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

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

应用方案D78040场扫描电路,偏转电流可达1.7Ap-p,可用于中小型显示器

D78040是一款场扫描电路&#xff0c;偏转电流可达1.7Ap-p&#xff0c;可用于中小型显示器。 二 特 点 1、有内置泵电源 2、垂直输出电路 3、热保护电路 4、偏转电流可达1.7Ap-p 三 基本参数 四 应用电路图 1、应用线路 2、PIN5脚输出波形如下&#xff1a;

多态的综合练习

题目要求 Animal类&#xff08;父类&#xff09; package www.jsu.com;public class Animal {private int age;private String color;public Animal() {}public Animal(int age, String color) {this.age age;this.color color;}public int getAge() {return age;}public voi…

数据安全之认识数据库防火墙

文章目录 一、什么是数据库防火墙二、数据库防火墙的主要功能三、数据库防火墙的工作原理四、数据库防火墙如何防护数据库免受SQL注入攻击五、数据库防火墙的部署方式六、数据库防火墙与网络防火墙的关系与区别七、数据库防火墙的应用场景 随着信息技术的快速发展&#xff0c;数…

ubuntu16如何使用高版本cmake

1.引言 最近在尝试ubuntu16.04下编译开源项目vsome&#xff0c;发现使用apt命令默认安装cmake的的版本太低。如下 最终得知&#xff0c;ubuntu16默认安装确实只能到3.5.1。解决办法只能是源码安装更高版本。 2.源码下载3.20 //定位到opt目录 cd /opt 下载 wget https://cmak…

蓝桥杯 - 受伤的皇后

解题思路&#xff1a; 递归 回溯&#xff08;n皇后问题的变种&#xff09; 在 N 皇后问题的解决方案中&#xff0c;我们是从棋盘的顶部向底部逐行放置皇后的&#xff0c;这意味着在任何给定时间&#xff0c;所有未来的行&#xff08;即当前行之下的所有行&#xff09;都还没…

昆仑万维将在4月17日正式开启“天工大模型3.0”的公测,并选择开源,音乐生成大模型“天工SkyMusic”开启内测

&#x1f989; AI新闻 &#x1f680; 昆仑万维发布全球最大MoE模型“天工3.0” 摘要&#xff1a;昆仑万维集团近日通过官方公众号宣布&#xff0c;将在4月17日正式开启“天工大模型3.0”的公测&#xff0c;并选择开源。此版本采用4000亿参数的混合专家模型(MoE)&#xff0c;被…

(echarts)vue中循环生成多个相同的echarts图表,但数据动态、第一次渲染失败问题

(echarts)vue中循环生成多个相同的echarts图表&#xff0c;但数据动态 效果&#xff1a; 代码&#xff1a; <!-- 动态图表 --> <el-row :gutter"20"><el-col v-for"(item,index) in echartsList" :key"index" :span"10&quo…

Android Studio学习7——常用控件view

Android控件 双击shift键——>搜索想要找的文件 Ctrlshift回车——>补全“&#xff1b;”号 CtrlX——>删除一行&#xff0c;只需把鼠标放在那一行 windows自带字体

如何备份极狐GitLab 信任域名证书

本文作者&#xff1a;徐晓伟 GitLab 是一个全球知名的一体化 DevOps 平台&#xff0c;很多人都通过私有化部署 GitLab 来进行源代码托管。极狐GitLab 是 GitLab 在中国的发行版&#xff0c;专门为中国程序员服务。可以一键式部署极狐GitLab。 本文主要讲述了如何使用极狐GitLa…

深信服:借助观测云实现全链路可观测性

导读 深信服科技股份有限公司 简称「深信服」&#xff08; Sangfor Technologies Inc. &#xff09;&#xff0c;是一家领先的网络安全和云计算解决方案提供商&#xff0c;致力于为全球客户提供高效、智能、安全的网络和云服务。随着公司业务的不断扩展&#xff0c;也面临着监…

超市销售数据-python数据分析项目

Python数据分析项目-基于Python的销售数据分析项目 文章目录 Python数据分析项目-基于Python的销售数据分析项目项目介绍数据分析结果导出数据查阅 数据分析内容哪些类别比较畅销?哪些商品比较畅销?不同门店的销售额占比哪个时间段是超市的客流高封期?查看源数据类型计算本月…

HTML - 请你谈一谈img标签图片和background背景图片的区别

难度级别&#xff1a;中级及以上 提问概率&#xff1a;65% 面试官当然不会问如何使用img标签或者background来加载一张图片&#xff0c;这些知识点都很基础&#xff0c;相信只要从事前端开发一小段时间以后&#xff0c;就可以轻松搞定加载图片…

stream使用

stream流式计算 在Java1.8之前还没有stream流式算法的时候&#xff0c;我们要是在一个放有多个User对象的list集合中&#xff0c;将每个User对象的主键ID取出&#xff0c;组合成一个新的集合&#xff0c;首先想到的肯定是遍历&#xff0c;如下&#xff1a; List<Long> u…

游戏引擎中的物理系统

一、物理对象与形状 1.1 对象 Actor 一般来说&#xff0c;游戏中的对象&#xff08;Actor&#xff09;分为以下四类&#xff1a; 静态对象 Static Actor动态对象 Dynamic Actor ---- 可能受到力/扭矩/冲量的影响检测器 TriggerKinematic Actor 运动学对象 ---- 忽略物理法则…

go: go.mod file not found in current directory or any parent directory.如何解决?

这个错误表明你正在执行 go get 命令&#xff0c;但是当前目录或任何父目录中都找不到 go.mod 文件。这可能是因为你的项目还没有使用 Go Modules 进行管理。 要解决这个问题&#xff0c;有几种方法&#xff1a; go mod init <module-name> 其中 <module-name>…

【力扣】11. 盛最多水的容器

11. 盛最多水的容器 题目描述 给定一个长度为 n 的整数数组 height 。有 n 条垂线&#xff0c;第 i 条线的两个端点是 (i, 0) 和 (i, height[i]) 。 找出其中的两条线&#xff0c;使得它们与 x 轴共同构成的容器可以容纳最多的水。 返回容器可以储存的最大水量。 说明&…

Python | Leetcode Python题解之第7题整数反转

题目&#xff1a; 题解&#xff1a; def reverse_better(self, x: int) -> int:y, res abs(x), 0# 则其数值范围为 [−2^31, 2^31 − 1]boundry (1<<31) -1 if x>0 else 1<<31while y ! 0:res res*10 y%10if res > boundry :return 0y //10return re…

数字化赋能农业:数字乡村促进农业现代化

随着信息技术的迅猛发展&#xff0c;数字化浪潮正以前所未有的速度席卷各行各业&#xff0c;农业领域也不例外。数字乡村战略作为推动农业现代化的重要手段&#xff0c;通过数字化技术的深度应用&#xff0c;为农业生产带来了革命性的变革。本文旨在探讨数字化如何赋能农业&…

用于HUD平视显示器的控制芯片:S2D13V40

一款利用汽车抬头显示技术用于HUD平视显示器的控制芯片:S2D13V40。HUD的全称是Head Up Display&#xff0c;即平视显示器&#xff0c;以前应用于军用飞机上&#xff0c;旨在降低飞行员需要低头查看仪表的频率。起初&#xff0c;HUD通过光学原理&#xff0c;将驾驶相关的信息投射…

隐私计算实训营第七讲-隐语SCQL的架构详细拆解

隐私计算实训营第七讲-隐语SCQL的架构详细拆解 文章目录 隐私计算实训营第七讲-隐语SCQL的架构详细拆解1.SCQL Overview1.1 多方数据分析场景1.2 多方数据分析技术路线1.2.1 TEE SQL方案1.2.2 MPC SQL方案 1.3 Secure Collaborative Query Language(SCQL)1.3.1 SCQL 系统组件1.…