说点不一样的 Spring 注册 IOC Bean 方式

简介: 说点不一样的 Spring 注册 IOC Bean 方式

每日一言


学习方法之多看:代码(别人的代码 or 源码)、规范(JEP、JSR)、论文(Paper) 等

前言


Spring 的出现是对广大 Java Web 开发者的福音,帮助我们解决了众多问题并且提供了很多便利

写此文章主要来说明对于 Spring IOC 容器中注册 Bean 的方式主要有哪几种

如果你以为我会告诉你

  • @Bean
  • @Import、@ImportSelector...
  • @Service、@Component...

很遗憾的说:格局小了兄弟...

虽然这篇文章是作者看 Mybatis Spring 集成源码之后,决定先水一篇探路的文章

但是!水也得水出高级感么不是,怎么不得让小伙伴有点收获,看点工作中不常用到的

注册 IOC 容器 Bean


除了上文中说到的注册 Bean 方式,文章主要使用 BeanDefinitionRegistryPostProcessor(以下简称:BD) 来进行 Bean 注册

名词可能没听说过,但是如果说它是BeanFactoryPostProcessor 后置处理器(以下简称BF)子类,应该就清楚很多了

整体的关系比较简洁,只是继承了 Spring 后置处理器,先看下接口定义是怎么介绍的

/**
 * Extension to the standard {@link BeanFactoryPostProcessor} SPI, allowing for
 * the registration of further bean definitions <i>before</i> regular
 * BeanFactoryPostProcessor detection kicks in. In particular,
 * BeanDefinitionRegistryPostProcessor may register further bean definitions
 * which in turn define BeanFactoryPostProcessor instances.
 *
 * @author Juergen Hoeller
 * @since 3.0.1
 * @see org.springframework.context.annotation.ConfigurationClassPostProcessor
 */
public interface BeanDefinitionRegistryPostProcessor extends BeanFactoryPostProcessor {

    /**
     * Modify the application context's internal bean definition registry after its
     * standard initialization. All regular bean definitions will have been loaded,
     * but no beans will have been instantiated yet. This allows for adding further
     * bean definitions before the next post-processing phase kicks in.
     * @param registry the bean definition registry used by the application context
     * @throws org.springframework.beans.BeansException in case of errors
     */
    void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) throws BeansException;
}

接口上面这一段让人看不懂的英文说的啥意思呢?

大致说的是当前接口是后置处理器的 SPI 扩展,会在后置处理器工作前定义更多的 Bean

因为 BD(可注册Bean接口) 继承了 BF(后置处理器) 接口,所以如果实现 BP 就需要实现两个方法

-- BeanFactoryPostProcessor
void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException;
-- BeanDefinitionRegistryPostProcessor
void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) throws BeansException;

mybatis 与 spring 源码集成上也是使用此方式,将 DAO 对象注册 Spring IOC 容器

postProcessBeanFactory

该方法主要对已注册 Spring Bean 的定义做出改变

postProcessBeanDefinitionRegistry

该方法主要是将 Bean 定义注册到 Spring IOC 容器中,具体的 Bean 细节体现入参 BeanDefinitionRegistry

BeanDefinitionRegistry

BeanDefinitionRegistry 是一个接口并实现了 AliasRegistry 接口,其中方法定义了对 Bean 的常用操作

// 向IOC容器中注册BeanDefinition实例
void registerBeanDefinition(String beanName, BeanDefinition beanDefinition)
            throws BeanDefinitionStoreException;
// 移除已存在的BeanDefinition实例
void removeBeanDefinition(String beanName) throws NoSuchBeanDefinitionException;
// 获取BeanDefinition实例
BeanDefinition getBeanDefinition(String beanName) throws NoSuchBeanDefinitionException;
// 判断beanName是否在IOC容器中
boolean containsBeanDefinition(String beanName);
// 获取IOC容器中所有实例名称
String[] getBeanDefinitionNames();
// 获取IOC容器中实例数量
int getBeanDefinitionCount();
// 判断beanName是否被占用
boolean isBeanNameInUse(String beanName);

postProcessBeanDefinitionRegistry 方法直接通过入参就可以完成 Bean 的注册、移除、判断等等操作

AliasRegistry 接口中定义的是 IOC 容器 Bean 别名管理,就不再一一介绍了

代码实战


创建一个 SpringBoot 的项目或者在已有项目测试都可以

@Component
public class BeanRegistryTest implements BeanDefinitionRegistryPostProcessor, ApplicationContextAware {

    private ApplicationContext applicationContext;

    @Override
    public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry beanDefinitionRegistry) throws BeansException {
        GenericBeanDefinition parentBeanDef = new GenericBeanDefinition();
        parentBeanDef.setBeanClass(BeanRegistry.class);
        beanDefinitionRegistry.registerBeanDefinition("beanRegistry", rootBeanDefinition);
    }

    @Override
    public void postProcessBeanFactory(ConfigurableListableBeanFactory configurableListableBeanFactory) throws BeansException {
        System.out.println("注册Bean对象 :: " + applicationContext.getBean("beanRegistry"));
    }

    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        this.applicationContext = applicationContext;
    }

    static class BeanRegistry { }
}

GenericBeanDefinition 是 BeanDefinition 接口通用实现,负责记录 Bean 对象信息

代码比较简单明了,为了方便在一个类中实现,通过以下步骤完成 Bean 注册测试

  1. 通过实现 ApplicationContextAware 接口赋值 ApplicationContext 对象
  2. BeanDefinitionRegistryPostProcessor 接口完成对象注册以及打印对应注册信息
  3. postProcessBeanDefinitionRegistry 方法中注册 BeanRegistry 到 Spring IOC 容器中
  4. postProcessBeanFactory 打印 Spring IOC 容器中的 BeanRegistry 对象信息

结言


由于作者水平有限, 欢迎大家能够反馈指正文章中错误不正确的地方, 感谢 🙏

小伙伴的喜欢就是对我最大的支持, 如果读了文章有所收获, 希望能够 点赞、评论、关注三连!

推荐阅读:

  1. 【强烈推荐】谨慎使用 JDK 8 新特性并行流 ParallelStream
  2. 【强烈推荐】一文快速掌握 Redisson 如何实现分布式锁原理
  3. 【大厂面试真题】JDK 线程池中如何不超最大线程数快速消费任务
  4. 【大厂面试真题】JDK 线程池如何保证核心线程不被销毁
相关文章
|
5天前
|
XML Java 数据格式
【SpringFramework】Spring IoC-基于XML的实现
本文主要讲解SpringFramework中IoC和DI相关概念,及基于XML的实现方式。
94 69
|
3天前
|
Java Spring 容器
【SpringFramework】Spring IoC-基于注解的实现
本文主要记录基于Spring注解实现IoC容器和DI相关知识。
35 21
|
9天前
|
设计模式 XML Java
【23种设计模式·全精解析 | 自定义Spring框架篇】Spring核心源码分析+自定义Spring的IOC功能,依赖注入功能
本文详细介绍了Spring框架的核心功能,并通过手写自定义Spring框架的方式,深入理解了Spring的IOC(控制反转)和DI(依赖注入)功能,并且学会实际运用设计模式到真实开发中。
【23种设计模式·全精解析 | 自定义Spring框架篇】Spring核心源码分析+自定义Spring的IOC功能,依赖注入功能
|
8天前
|
存储 Java Spring
【Spring】获取Bean对象需要哪些注解
@Conntroller,@Service,@Repository,@Component,@Configuration,关于Bean对象的五个常用注解
|
8天前
|
存储 Java 应用服务中间件
【Spring】IoC和DI,控制反转,Bean对象的获取方式
IoC,DI,控制反转容器,Bean的基本常识,类注解@Controller,获取Bean对象的常用三种方式
|
14天前
|
XML Java 数据格式
Spring容器Bean之XML配置方式
通过对以上内容的掌握,开发人员可以灵活地使用Spring的XML配置方式来管理应用程序的Bean,提高代码的模块化和可维护性。
51 6
|
15天前
|
XML Java 数据格式
🌱 深入Spring的心脏:Bean配置的艺术与实践 🌟
本文深入探讨了Spring框架中Bean配置的奥秘,从基本概念到XML配置文件的使用,再到静态工厂方式实例化Bean的详细步骤,通过实际代码示例帮助读者更好地理解和应用Spring的Bean配置。希望对你的Spring开发之旅有所助益。
79 3
|
29天前
|
存储 缓存 Java
Spring面试必问:手写Spring IoC 循环依赖底层源码剖析
在Spring框架中,IoC(Inversion of Control,控制反转)是一个核心概念,它允许容器管理对象的生命周期和依赖关系。然而,在实际应用中,我们可能会遇到对象间的循环依赖问题。本文将深入探讨Spring如何解决IoC中的循环依赖问题,并通过手写源码的方式,让你对其底层原理有一个全新的认识。
51 2
|
29天前
|
安全 Java 开发者
Spring容器中的bean是线程安全的吗?
Spring容器中的bean默认为单例模式,多线程环境下若操作共享成员变量,易引发线程安全问题。Spring未对单例bean做线程安全处理,需开发者自行解决。通常,Spring bean(如Controller、Service、Dao)无状态变化,故多为线程安全。若涉及线程安全问题,可通过编码或设置bean作用域为prototype解决。
34 1
|
5月前
|
XML Java 数据格式
Spring5入门到实战------7、IOC容器-Bean管理XML方式(外部属性文件)
这篇文章是Spring5框架的实战教程,主要介绍了如何在Spring的IOC容器中通过XML配置方式使用外部属性文件来管理Bean,特别是数据库连接池的配置。文章详细讲解了创建属性文件、引入属性文件到Spring配置、以及如何使用属性占位符来引用属性文件中的值。
Spring5入门到实战------7、IOC容器-Bean管理XML方式(外部属性文件)