2022-08-23  阅读(41)
版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。 本文链接:https://www.skjava.com/series/article/1803183842

Spring 在 bean 初始化时进行三个检测扩展,也就是说我们可以对 bean 进行三个不同的定制化处理,前面两篇博客 【死磕 Spring】----- IOC 之 深入分析 Aware 接口 和 【死磕 Spring】----- IOC 之 深入分析 BeanPostProcessor 已经分析了 Aware 接口族 和 BeanPostProcessor 接口,这篇分析 InitializingBean 接口和 init-method 方法。

InitializingBean

Spring 的 InitializingBean 接口为 bean 提供了定义初始化方法的方式,它仅包含了一个方法:afterPropertiesSet()

            public interface InitializingBean {
            
             /**
                * 该方法在 BeanFactory 设置完了所有属性之后被调用
                * 该方法允许 bean 实例设置了所有 bean 属性时执行初始化工作,如果该过程出现了错误则需要抛出异常
              */
             void afterPropertiesSet() throws Exception;
            }

Spring 在完成实例化后,设置完所有属性,进行 “Aware 接口” 和 “BeanPostProcessor 前置处理”之后,会接着检测当前 bean 对象是否实现了 InitializingBean 接口,如果是,则会调用其 afterPropertiesSet() 进一步调整 bean 实例对象的状态。

            public class InitializingBeanTest implements InitializingBean {
            
                private String name;
            
                @Override
                public void afterPropertiesSet() throws Exception {
                    System.out.println("InitializingBeanTest initializing...");
            
                    this.name = "chenssy 2 号";
                }
            
                public String getName() {
                    return name;
                }
            
                public void setName(String name) {
                    this.name = name;
                }
            }
            
            // 配置项
            <bean id="initializingBeanTest" class="org.springframework.core.test.InitializingBeanTest">
                <property name="name" value="chenssy 1 号"/>
            </bean>
            
            // 测试代码
            InitializingBeanTest test = (InitializingBeanTest) factory.getBean("initializingBeanTest");
            System.out.println("name :" + test.getName());

执行结果:

202202131351014801.png

在这个示例中改变了 InitializingBeanTest 示例的 name 属性,也就是说 在 afterPropertiesSet() 中我们是可以改变 bean 的属性的,这相当于 Spring 容器又给我们提供了一种可以改变 bean 实例对象的方法。

上面提到 bean 初始化阶段(initializeBean() ) Spring 容器会主动检查当前 bean 是否已经实现了 InitializingBean 接口,如果实现了则会掉用其 afterPropertiesSet() ,这个主动检查、调用的动作是由 invokeInitMethods() 来完成的。

                protected void invokeInitMethods(String beanName, final Object bean, @Nullable RootBeanDefinition mbd)
                        throws Throwable {
            
                    // 是否实现 InitializingBean
                    // 如果实现了 InitializingBean 接口,则只掉调用bean的 afterPropertiesSet()
                    boolean isInitializingBean = (bean instanceof InitializingBean);
                    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((PrivilegedExceptionAction<Object>) () -> {
                                    ((InitializingBean) bean).afterPropertiesSet();
                                    return null;
                                }, getAccessControlContext());
                            }
                            catch (PrivilegedActionException pae) {
                                throw pae.getException();
                            }
                        }
                        else {
                            // 直接调用 afterPropertiesSet()
                            ((InitializingBean) bean).afterPropertiesSet();
                        }
                    }
            
                    if (mbd != null && bean.getClass() != NullBean.class) {
                        // 判断是否指定了 init-method(),
                        // 如果指定了 init-method(),则再调用制定的init-method
                        String initMethodName = mbd.getInitMethodName();
                        if (StringUtils.hasLength(initMethodName) &&
                                !(isInitializingBean && "afterPropertiesSet".equals(initMethodName)) &&
                                !mbd.isExternallyManagedInitMethod(initMethodName)) {
                            // 利用反射机制执行
                            invokeCustomInitMethod(beanName, bean, mbd);
                        }
                    }
                }

首先检测当前 bean 是否实现了 InitializingBean 接口,如果实现了则调用其 afterPropertiesSet(),然后再检查是否也指定了 init-method(),如果指定了则通过反射机制调用指定的 init-method()

虽然该接口为 Spring 容器的扩展性立下了汗马功劳,但是如果真的让我们的业务对象来实现这个接口就显得不是那么的友好了,Spring 的一个核心理念就是无侵入性,但是如果我们业务类实现这个接口就显得 Spring 容器具有侵入性了。所以 Spring 还提供了另外一种实现的方式:init-method 方法

init-method()

在分析分析 <bean> 标签解析过程中我们提到了有关于 init-method 属性 (【死磕 Spring】----- IOC 之解析Bean:解析 bean 标签(二)),该属性用于在 bean 初始化时指定执行方法,可以用来替代实现 InitializingBean 接口。

            public class InitializingBeanTest {
            
                private String name;
            
                public String getName() {
                    return name;
                }
            
                public void setName(String name) {
                    this.name = name;
                }
            
                public void setOtherName(){
                    System.out.println("InitializingBeanTest setOtherName...");
            
                    this.name = "chenssy 3 号";
                }
            }
            
            // 配置文件
            <bean id="initializingBeanTest" class="org.springframework.core.test.InitializingBeanTest"
                    init-method="setOtherName">
                <property name="name" value="chenssy 1 号"/>
            </bean>

执行结果:

202202131351022642.png

完全可以达到和 InitializingBean 一样的效果,而且在代码中我们没有看到丝毫 Spring 侵入的现象。所以通过 init-method 我们可以使用业务对象中定义的任何方法来实现 bean 实例对象的初始化定制化,而不再受制于 InitializingBean的 afterPropertiesSet()。同时我们可以使用 <beans> 标签的 default-init-method 属性来统一指定初始化方法,这样就省了需要在每个 <bean> 标签中都设置 init-method 这样的繁琐工作了。比如在 default-init-method 规定所有初始化操作全部以 initBean() 命名。如下:

202202131351032333.png

invokeInitMethods() 中,我们知道 init-method 指定的方法会在 afterPropertiesSet() 之后执行,如果 afterPropertiesSet() 中出现了异常,则 init-method 是不会执行的,而且由于 init-method 采用的是反射执行的方式,所以 afterPropertiesSet() 的执行效率一般会高些,但是并不能排除我们要优先使用 init-method,主要是因为它消除了 bean 对 Spring 的依赖,Spring 没有侵入到我们业务代码,这样会更加符合 Spring 的理念。诚然,init-method 是基于 xml 配置文件的,就目前而言,我们的工程几乎都摒弃了配置,而采用注释的方式,那么 @PreDestory 可能适合你,当然这个注解我们后面分析。

至此,InitializingBean 和 init-method 已经分析完毕了,对于DisposableBean 和 destory-method,他们和 init 相似,这里就不做阐述了。


Java 面试宝典是大明哥全力打造的 Java 精品面试题,它是一份靠谱、强大、详细、经典的 Java 后端面试宝典。它不仅仅只是一道道面试题,而是一套完整的 Java 知识体系,一套你 Java 知识点的扫盲贴。

它的内容包括:

  • 大厂真题:Java 面试宝典里面的题目都是最近几年的高频的大厂面试真题。
  • 原创内容:Java 面试宝典内容全部都是大明哥原创,内容全面且通俗易懂,回答部分可以直接作为面试回答内容。
  • 持续更新:一次购买,永久有效。大明哥会持续更新 3+ 年,累计更新 1000+,宝典会不断迭代更新,保证最新、最全面。
  • 覆盖全面:本宝典累计更新 1000+,从 Java 入门到 Java 架构的高频面试题,实现 360° 全覆盖。
  • 不止面试:内容包含面试题解析、内容详解、知识扩展,它不仅仅只是一份面试题,更是一套完整的 Java 知识体系。
  • 宝典详情:https://www.yuque.com/chenssy/sike-java/xvlo920axlp7sf4k
  • 宝典总览:https://www.yuque.com/chenssy/sike-java/yogsehzntzgp4ly1
  • 宝典进展:https://www.yuque.com/chenssy/sike-java/en9ned7loo47z5aw

目前 Java 面试宝典累计更新 400+ 道,总字数 42w+。大明哥还在持续更新中,下图是大明哥在 2024-12 月份的更新情况:

想了解详情的小伙伴,扫描下面二维码加大明哥微信【daming091】咨询

同时,大明哥也整理一套目前市面最常见的热点面试题。微信搜[大明哥聊 Java]或扫描下方二维码关注大明哥的原创公众号[大明哥聊 Java] ,回复【面试题】 即可免费领取。

阅读全文