我的编程空间,编程开发者的网络收藏夹
学习永远不晚

详细SpringBoot生命周期接口的使用

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

详细SpringBoot生命周期接口的使用

一 背景

最近在做一个项目启动时加载配置到SpringBoot容器中的功能,看到了Spring中有很多在容器初始化时的接口,这些接口或注解包括InitializingBean、@PostConstruct、SmartInitializingSingleton、BeanPostProcess等等,这么多都可以在初始化时使用,但是他们有什么区别呢,下面就来说说他们之间的区别

二 SpringBoot 生命周期接口

  • @PostConstruct

    这个注解在实际的开发中有较多的用到

    @Component
    public class TestP {
    
        @PostConstruct
        public void test() {
            System.out.println("@PostConstruct");
        }
    

    这样在容器启动过程中就回执行打印,看起来他像是对象的构造方法,其实他的作用是,当一个对象A中存在@Autowire修饰的依赖B时,正常来说,对象会先执行自己的构造方法,然后再去注入依赖,但是我们现在有一种情况,在对象实例化时,要执行构造方法,但是构造方法中用到依赖B,这个时候用 @PostConstruct就能解决这个问题。

    public class TestP implements InitializingBean, SmartInitializingSingleton {
        @Autowired
        Context context;
        ```
        public TestP () {
            System.out.println(context);
        }
        ```
        @PostConstruct
        public void test() {
            System.out.println(context);
            System.out.println("@PostConstruct");
        }
    

    输出

 null
 com.example.demo1.bean.Context@4f96a58
 @PostConstruct

  • InitializingBean

    他提供bean初始化的回调处理功能,看下这个接口的源码

    protected void invokeInitMethods(String beanName, final Object bean, RootBeanDefinition mbd) throws Throwable {
     boolean isInitializingBean = (bean instanceof InitializingBean);
    
    //判断该bean是否实现了实现了InitializingBean接口,如果实现了InitializingBean接口,则只掉调用bean的afterPropertiesSet方法
     if (isInitializingBean && (mbd == null || !mbd.isExternallyManagedInitMethod("afterPropertiesSet"))) {
         if (logger.isDebugEnabled()) {
             logger.debug("Invoking afterPropertiesSet() on bean with name '" + beanName + "'");
         }
    
         if (System.getSecurityManager() != null) {
             try {
                 AccessController.doPrivileged(new PrivilegedExceptionAction<Object>() {
                     public Object run() throws Exception {
                         //直接调用afterPropertiesSet
                         ((InitializingBean) bean).afterPropertiesSet();
                         return null;
                     }
                 },getAccessControlContext());
             } catch (PrivilegedActionException pae) {
                 throw pae.getException();
             }
         }                
         else {
             //直接调用afterPropertiesSet
             ((InitializingBean) bean).afterPropertiesSet();
         }
     }
     if (mbd != null) {
         String initMethodName = mbd.getInitMethodName();
         //判断是否指定了init-method方法,如果指定了init-method方法,则再调用制定的init-method
         if (initMethodName != null && !(isInitializingBean && "afterPropertiesSet".equals(initMethodName)) &&
                 !mbd.isExternallyManagedInitMethod(initMethodName)) {
             //进一步查看该方法的源码,可以发现init-method方法中指定的方法是通过反射实现
             invokeCustomInitMethod(beanName, bean, mbd);
         }
     }
    

    可以看到,只要实现了这个接口的bean都会执行里面的afterPropertiesSet方法,那他和 @PostConstruct有什么区别呢,区别是他们的生命周期排序不同,@PostConstruct是对单个Bean实例化时使用,而InitializingBean是在所有spring bean实例化后对bean进行处理,⼤致的步骤是这样的

    实例化bean,这⾥会调⽤构造⽅法

    填充属性,就是依赖注⼊

    初始化bean,

    • 调⽤后置处理器,其中会执⾏@PostConstruct注解⽅法
    • 执⾏bean的⽣命周期中的初始化回调⽅法,也就是InitializingBean接⼝的afterPropertiesSet()⽅法
  • BeanPostProcess

    这个接口主要是对注册的bean中的属性进行初始化时的修改

    @Component
    public class BeanPostTest implements BeanPostProcessor {
        @Override
        public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
            System.out.println("执行1--------------- - " + bean.getClass().getName() + " - " + beanName);
            return null;
        }
        @Override
        public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
            System.out.println("执行2--------------- - " + bean.getClass().getName() + " - " + beanName);
            return null;
        }
    }
    

    他和InitializingBean区别一个是执行顺序不同

他有两个方法分别在在InitializingBean执行前执行后执行,第二个区别就是InitializingBean是一个可以进一步调整bean的实例的接口,不过并不是每个类都会来执行这个接口方法,这个接口只针对当前实现类,而BeanPostProcess是针对所有bean的,每一个bean被注册,都会被执行一次这两个方法

  • SmartInitializingSingleton

    这个是在spring 4.1版本才推出的接口,他的执行时时机是在单例预实例化阶段结束时调用,并保证已经创建了所有常规单例bean,所以他的执行顺序是比较靠后的,考虑到一些bean的注册及修改使用SmartInitializingSingleton是比较稳妥的一种方式

     ```
     @Component
     public class GetuiAccountConfigInit implements SmartInitializingSingleton  {
    
    
         @Override
         public void afterSingletonsInstantiated() {
    
         }
     }
     ```
    
  • Commandlinerunner 这个接口是springBoot的接口,他是在所有bean都加载后才会执行的,如果实现这个接口,可以很好的在启动时初始化资源,因为所有的bean都可以使用了

     @Component
     public class Runner implements CommandLineRunner {
         @Override
         public void run(String... args) throws Exception {
             System.out.println("执行初始化");
         }
     }
    

    如果我们要执行的程序有顺序要求,还可以使用@Order注解

     @Component
     @Order(1)
     public class OrderRunner1 implements CommandLineRunner {
         @Override
         public void run(String... args) throws Exception {
             System.out.println("The OrderRunner1 start to initialize ...");
         }
     }
     @Component
     @Order(2)
     public class OrderRunner1 implements CommandLineRunner {
         @Override
         public void run(String... args) throws Exception {
             System.out.println("The OrderRunner2 start to initialize ...");
         }
     }
    

三 后记

Spring由于设计上比较灵活所以留了很多接口,让开发人员进行拓展,这本身是一个很好的学习借鉴的经验,现在大部分的开发使用的都是spring的框架,这就要求我们做一些设计时要了解框架的特性,才能进行更好的设计,上面的几个接口是相对来说比较常用的接口,里面的技术细节也值得推敲,希望对大家有所帮助。

到此这篇关于详细SpringBoot生命周期接口的使用的文章就介绍到这了,更多相关SpringBoot生命周期接口内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

详细SpringBoot生命周期接口的使用

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

Android Activity的生命周期详细介绍

Android Activity的生命周期详细介绍 生命周期描述的是一个类从创建(new出来)到死亡(垃圾回收)的过程中会执行的方法。在这个过程中,会针对不同的生命阶段调用不同的的方法。 Activity是Android中四大组
2022-06-06

基于Android Service 生命周期的详细介绍

Service概念及用途: Android中的服务,它与Activity不同,它是不能与用户交互的,不能自己启动的,运行在后台的程序,如果我们退出应用时,Service进程并没有结束,它仍然在后台运行,那我们什么时候会用到Service呢?
2022-06-06

C++临时性对象的生命周期详细解析

临时性对象的被摧毁,应该是对完整表达式(full-expression)求值过程中的最后一个步骤。该完整表达式造成临时对象的产生
2022-11-15

JVM类加载机制及生命周期的详细介绍

这篇文章主要讲解了“JVM类加载机制及生命周期的详细介绍”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“JVM类加载机制及生命周期的详细介绍”吧!一.目标:1.什么是类的加载?2.类的生命周期
2023-06-02

react的生命周期函数怎么使用

这篇文章主要介绍“react的生命周期函数怎么使用”,在日常操作中,相信很多人在react的生命周期函数怎么使用问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”react的生命周期函数怎么使用”的疑惑有所帮助!
2023-07-04

微信小程序开发中生命周期的详细介绍

生命周期是指一个对象从创建→>运行>销毁的整个阶段,强调的是一个时间段,文中介绍了小程序中组件的生命周期,需要的朋友可以参考下
2023-05-13

React中state属性和生命周期的使用

这篇文章主要介绍了React中state属性和生命周期的使用说明,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
2023-02-12

AndroidActivity的生命周期与加载模式超详细图文解析

这篇文章主要介绍了AndroidActivity的生命周期与加载模式,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
2023-02-28

vue的混入可以使用生命周期吗

vue的混入可以使用生命周期。mixins(混入)中的生命周期会与引入mixins的组件的生命周期整合在一起调用,而且mixins中的生命周期函数会比引入mixins的组件调用的快。需要注意,多个mixins的生命周期会融合到一起运行,但是同名属性、同名方法无法融合,可能会导致冲突或覆盖。
2023-05-14

vue的混入可不可以使用生命周期

这篇文章主要介绍“vue的混入可不可以使用生命周期”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“vue的混入可不可以使用生命周期”文章能帮助大家解决问题。vue的混入可以使用生命周期。mixins(
2023-07-04

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录