首页 > 其他分享 >SpringBoot中操作Bean的生命周期的方法

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

时间:2024-01-18 11:23:49浏览次数:19  
标签:生命周期 SpringBoot springframework class Bean org import public

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

路人 路人甲Java 2024-01-17 19:17 发表于上海

引言

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

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

1. InitializingBeanDisposableBean 接口:

在某些环境或特定的约束下,如果您想避免使用 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 定义的 initMethoddestroyMethod

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

  • 在 Bean 定义中,可以通过 initMethoddestroyMethod 属性指定初始化和销毁方法。
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 接口提供了两个方法:postProcessBeforeInitializationpostProcessAfterInitialization,分别在 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 接口:![图片](data:image/svg+xml,%3C%3Fxml version='1.0' encoding='UTF-8'%3F%3E%3Csvg width='1px' height='1px' viewBox='0 0 1 1' version='1.1' xmlns='http://www.w3.org/2000/svg' xmlns:xlink='http://www.w3.org/1999/xlink'%3E%3Ctitle%3E%3C/title%3E%3Cg stroke='none' stroke-width='1' fill='none' fill-rule='evenodd' fill-opacity='0'%3E%3Cg transform='translate(-249.000000, -126.000000)' fill='%23FFFFFF'%3E%3Crect x='249' y='126' width='1' height='1'%3E%3C/rect%3E%3C/g%3E%3C/g%3E%3C/svg%3E)

  • 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. 实现 ApplicationContextAwareBeanNameAware 接口:

  • 这些接口允许 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. 使用 EnvironmentAwareResourceLoaderAware 接口:

  • 这些接口允许 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. 使用 @OrderOrdered 接口:

  • 这些用于定义 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 应用更加健壮、灵活和高效。

标签:生命周期,SpringBoot,springframework,class,Bean,org,import,public
From: https://www.cnblogs.com/hefeng2014/p/17972128

相关文章

  • Sa-Token介绍与SpringBoot环境下使用
    个人博客:无奈何杨(wnhyang)个人语雀:wnhyang共享语雀:在线知识共享Github:wnhyang-Overview官网:Sa-Token一个轻量级Java权限认证框架,让鉴权变得简单、优雅!介绍Sa-Token是一个轻量级Java权限认证框架,主要解决:登录认证、权限认证、单点登录、OAuth2.0、分布式Session会话......
  • 0.o?让我看看怎么个事儿之SpringBoot自动配置
    学习SpringBoot自动配置之前我们需要一些前置知识点:Java注解,看完就会用学会@ConfigurationProperties月薪过三千不是银趴~是@Import!@Conditional+@Configuration有没有搞头?首先我们提出2个问题:SpringBoot是干什么的?是用来简化Spring原生的复杂的xml配置的进阶框架......
  • SpringBoot+MybatisPlus+dynamic-datasources实现连接Postgresql和mysql多数据源
    场景dynamic-datasource-spring-boot-starter实现动态数据源Mysql和Sqlserver:https://blog.csdn.net/BADAO_LIUMANG_QIZHI/article/details/117356693SpringBoot中整合MybatisPlus快速实现Mysql增删改查和条件构造器:https://blog.csdn.net/BADAO_LIUMANG_QIZHI/article/detail......
  • SpringBoot中整合MybatisPlus快速实现Mysql增删改查和条件构造器
    场景Mybatis-Plus(简称MP)是一个Mybatis的增强工具,只是在Mybatis的基础上做了增强却不做改变,MyBatis-Plus支持所有Mybatis原生的特性,所以引入Mybatis-Plus不会对现有的Mybatis构架产生任何影响。MyBatis增强工具包,简化CRUD操作。启动加载XML配置时注入单表SQL操作,为简......
  • Springboot项目配置多数据源,然后任意切换
    数据库信息spring.datasource.url=jdbc:mysql://127.0.0.1:3306/xxl_job_test?useUnicode=true&characterEncoding=UTF-8&autoReconnect=true&serverTimezone=Asia/Shanghaispring.datasource.username=rootspring.datasource.password=rootspring.datasource.sec......
  • 基于SpringBoot+Vue的校园招聘系统设计实现(源码+lw+部署文档+讲解等)
    (文章目录)前言:heartpulse:博主介绍:✌全网粉丝10W+,CSDN特邀作者、博客专家、CSDN新星计划导师、全栈领域优质创作者,博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java、小程序技术领域和毕业项目实战✌:heartpulse:......
  • SpringBoot项目从0到1配置logback日志打印
    转载自:SpringBoot项目从0到1配置logback日志打印,作者sum墨一、写文背景我们在写后端项目的时候,日志打印是必需的。支持SpringBoot项目的日志框架一般有log4j、logback,这二者各有优劣,这里就不展开对比了。我们项目中常用的是logback框架,该框架主要是一个logback-spring.xml配置......
  • SpringBoot使用jwt实现接口带令牌访问
    1、 添加依赖<dependency><groupId>com.auth0</groupId><artifactId>java-jwt</artifactId><version>${jwt.version}</version></dependency>2、先定义一个用户实体@Data@TableName("t_user")publicclass......
  • spring boot2 bean和代理
    众所周知,我们可以从applicationContext根据name来获取bean,我曾一度以为bean就是bean自己,spring帮我们new出来的一个class对象,但当我读到下图这句话的时候,有点懵,getBean得到的为啥是代理对象??? 不过又一想,方法上有Transactional注解,Transactional会帮你做一些事务的commit,rollbac......
  • Springboot3+Vue3在进行WebSocket通讯时出现No mapping for GET或者是404
    参考:在SpringBoot中整合、使用WebSocket-spring中文网(springdoc.cn)===============================原代码(此时前端访问后端,后端会出现:NomappingforGET/wspath)前端相关代码:letsocket:WebSocket|null=nullconstsocketURL=`ws://127.0.0.1:8084/w......