Spring揭秘:ImportBeanDefinitionRegistrar应用场景及实现原理!

news2025/1/26 15:38:21

Spring揭秘:ImportBeanDefinitionRegistrar应用场景及实现原理! - 程序员古德

内容概念

ImportBeanDefinitionRegistrar接口提供了强大的动态注册Bean的能力,它允许开发者在Spring容器初始化时,灵活地根据特定条件或需求来添加或修改Bean定义,从而实现更为精细的控制和扩展性。这是构建可扩展框架、插件系统或处理复杂配置场景的利器。

核心概念

ImportBeanDefinitionRegistrar是Spring框架中一个非常强大的接口,它允许在运行时动态地向Spring容器中注册Bean定义,这特性在一些需要动态扩展、插件化或者编程式配置Spring应用的场景中特别有用。

模拟一个业务案例,假如,有一个电商平台,平台支持多种支付方式,比如支付宝、微信支付、银联支付等,每种支付方式都有自己的配置参数和实现逻辑,而且这些支付方式可能会随着业务的发展不断增加或变更。

传统的做法可能是为每种支付方式编写一个配置类,然后在主配置类中使用@Import注解将这些配置类静态地导入到Spring容器中,但这种方式不够灵活,每次增加新的支付方式时都需要修改主配置类,并且需要重启应用才能生效。

类似这样的场景,就非常适合用ImportBeanDefinitionRegistrar来解决,可以创建一个实现了ImportBeanDefinitionRegistrar接口的类,比如叫做PaymentRegistrar,在这个类中,可以编写逻辑来动态地扫描和识别所有可用的支付方式,并为每种支付方式创建一个对应的Bean定义,然后注册到Spring容器中。

可以在PaymentRegistrarregisterBeanDefinitions方法中编写逻辑,实现思路大概如下:

  1. 扫描指定路径下的支付方式实现类。
  2. 对于每个找到的支付方式实现类,创建一个对应的Bean定义,并设置必要的属性,比如支付URL、密钥等。
  3. 将这些Bean定义注册到传入的BeanDefinitionRegistry中。

最后,在主配置类中使用@Import注解将PaymentRegistrar导入到Spring容器中,这样,当应用启动时,Spring会自动调用PaymentRegistrarregisterBeanDefinitions方法,从而动态地加载和注册所有可用的支付方式。

这种方式,可以在不修改主配置类和重启应用的情况下,灵活地添加、删除或修改支付方式。这对于快速响应业务需求变化和降低维护成本非常有帮助。

核心案例

下面是一个简单的Java例子,演示了如何使用ImportBeanDefinitionRegistrar来动态注册Bean定义,在例子中,创建一个简单的服务接口GreetingService,并提供两个实现类EnglishGreetingServiceSpanishGreetingService,然后使用ImportBeanDefinitionRegistrar来动态地注册这些服务,并通过客户端代码来调用它们,如下代码:

首先,定义服务接口和实现类:

// GreetingService.java  
public interface GreetingService {  
    String sayGreeting();  
}  
  
// EnglishGreetingService.java  
public class EnglishGreetingService implements GreetingService {  
    @Override  
    public String sayGreeting() {  
        return "Hello!";  
    }  
}  
  
// SpanishGreetingService.java  
public class SpanishGreetingService implements GreetingService {  
    @Override  
    public String sayGreeting() {  
        return "¡Hola!";  
    }  
}

接下来,创建实现了ImportBeanDefinitionRegistrar接口的类,用于动态注册Bean定义:

// GreetingServiceRegistrar.java  
import org.springframework.beans.factory.support.BeanDefinitionRegistry;  
import org.springframework.beans.factory.support.GenericBeanDefinition;  
import org.springframework.context.annotation.ImportBeanDefinitionRegistrar;  
import org.springframework.core.type.AnnotationMetadata;  
  
public class GreetingServiceRegistrar implements ImportBeanDefinitionRegistrar {  
  
    @Override  
    public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {  
        // 动态注册 EnglishGreetingService  
        GenericBeanDefinition englishGreetingServiceDefinition = new GenericBeanDefinition();  
        englishGreetingServiceDefinition.setBeanClassName(EnglishGreetingService.class.getName());  
        registry.registerBeanDefinition("englishGreetingService", englishGreetingServiceDefinition);  
  
        // 动态注册 SpanishGreetingService  
        GenericBeanDefinition spanishGreetingServiceDefinition = new GenericBeanDefinition();  
        spanishGreetingServiceDefinition.setBeanClassName(SpanishGreetingService.class.getName());  
        registry.registerBeanDefinition("spanishGreetingService", spanishGreetingServiceDefinition);  
    }  
}

现在,需要在Spring配置中使用@Import注解来导入GreetingServiceRegistrar

// AppConfig.java  
import org.springframework.context.annotation.Configuration;  
import org.springframework.context.annotation.Import;  
  
@Configuration  
@Import(GreetingServiceRegistrar.class)  
public class AppConfig {  
    // 其他配置...  
}

最后,编写客户端代码来调用动态注册的Bean:

// Application.java  
import org.springframework.context.ApplicationContext;  
import org.springframework.context.annotation.AnnotationConfigApplicationContext;  
  
public class Application {  
  
    public static void main(String[] args) {  
        ApplicationContext context = new AnnotationConfigApplicationContext(AppConfig.class);  
  
        // 获取并调用 EnglishGreetingService  
        GreetingService englishService = context.getBean("englishGreetingService", GreetingService.class);  
        System.out.println(englishService.sayGreeting()); // 输出: Hello!  
  
        // 获取并调用 SpanishGreetingService  
        GreetingService spanishService = context.getBean("spanishGreetingService", GreetingService.class);  
        System.out.println(spanishService.sayGreeting()); // 输出: ¡Hola!  
    }  
}

在上面的代码中,使用了AnnotationConfigApplicationContext来创建一个Spring应用上下文,并指定了配置类AppConfig,然后,使用context.getBean()方法来获取动态注册的Bean,并调用它们的方法来输出问候语,输出结果应该是分别打印出"Hello!“和”¡Hola!"。

核心API

ImportBeanDefinitionRegistrar 接口允许开发者在运行时动态地向 Spring 应用程序上下文中注册 Bean 定义,这个接口通常与 @Import 注解结合使用,当 Spring 容器扫描到带有 @Import 注解的类时,会调用实现了 ImportBeanDefinitionRegistrar 接口的类的相关方法,ImportBeanDefinitionRegistrar 接口中只有一个方法,它的核心方法以及含义如下:
registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry),参数:

1、importingClassMetadata:提供有关正在导入的类的元数据信息,如类名、注解等。

2、registry:用于注册 Bean 定义的 BeanDefinitionRegistry,可以使用这个注册中心来添加、移除或修改 Bean 定义。

该方法的用途:

  1. 该方法是 ImportBeanDefinitionRegistrar 接口的核心,它允许开发者在 Spring 容器初始化过程中动态地添加或修改 Bean 定义。
  2. 通过 BeanDefinitionRegistry,可以创建新的 BeanDefinition 对象,并使用 registerBeanDefinition 方法将其注册到容器中。
  3. 还可以使用其他 BeanDefinitionRegistry 的方法来修改已存在的 Bean 定义或查询容器中的 Bean 定义。

技术原理

ImportBeanDefinitionRegistrar接口的实现类会在Spring容器解析到带有@Import注解的配置类时被调用,从而允许开发者在容器初始化过程中动态地添加、修改或删除Bean定义。

实现原理

  1. @Import注解的解析
    当Spring容器解析到带有@Import注解的类时,它会查看该注解所引用的类,如果这些类实现了ImportBeanDefinitionRegistrar接口,Spring容器就会创建这些类的实例,并调用它们的registerBeanDefinitions方法。
  2. registerBeanDefinitions方法的调用
    registerBeanDefinitions方法接收两个参数:一个是AnnotationMetadata,它包含了关于正在被处理的注解类的元数据(如类名、方法、其他注解等);另一个是BeanDefinitionRegistry,它是一个允许操作容器中Bean定义的注册表。
  3. 动态注册Bean定义
    registerBeanDefinitions方法内部,开发者可以编写自定义逻辑来创建BeanDefinition对象(这些对象描述了如何创建Bean实例),并使用BeanDefinitionRegistry将它们注册到Spring容器中,注册过程可以基于传入的AnnotationMetadata来做出决策。

工作流程

  1. 扫描和解析注解
    Spring容器在启动时会扫描指定的包路径,查找并解析带有特定注解(如@Component, @Service, @Repository, @Controller, @Configuration等)的类,当遇到@Import注解时,它会特别处理。
  2. 处理@Import注解
    对于每个@Import注解,Spring会查看其值(即要导入的类),并检查这些类是否实现了ImportBeanDefinitionRegistrar接口,如果实现了,就会实例化这些类,并准备调用它们的registerBeanDefinitions方法。
  3. 执行自定义注册逻辑
    对于每个实现了ImportBeanDefinitionRegistrar的类,Spring会调用其registerBeanDefinitions方法,在这个方法中,开发者可以编写任意逻辑来创建和注册Bean定义,这通常涉及到创建BeanDefinition对象(如GenericBeanDefinition),设置其属性(如bean类名、作用域、依赖等),然后使用BeanDefinitionRegistryregisterBeanDefinition方法将其注册到容器中。
  4. 完成容器初始化
    在调用了所有ImportBeanDefinitionRegistrar实现类的registerBeanDefinitions方法后,Spring容器会继续其初始化过程,包括创建和初始化所有已注册的Bean实例。

核心总结

Spring揭秘:ImportBeanDefinitionRegistrar应用场景及实现原理! - 程序员古德

优点在于灵活性高,允许开发者在Spring容器初始化时,根据特定条件或逻辑动态地添加、修改Bean定义,实现更细粒度的控制,对于编写框架代码或需要动态扩展功能的应用来说非常有用。

但是,由于是在运行时动态注册Bean,可能会增加容器的启动时间和复杂性,推荐,在确实需要动态注册Bean的场景下使用,如插件系统、动态数据源等。

关注我,每天学习互联网编程技术 - 程序员古德

END!
END!
END!

往期回顾

精品文章

Spring揭秘:@import注解应用场景及实现原理!

Java并发基础:原子类之AtomicMarkableReference全面解析!

Java并发基础:concurrent Flow API全面解析

Java并发基础:CopyOnWriteArraySet全面解析

Java并发基础:ConcurrentSkipListMap全面解析

精彩视频

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

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

相关文章

请说说你对Vue模板编译的理解

Vue模板编译是Vue.js框架的核心之一,它负责将Vue模板转换成渲染函数,从而实现模板的解析和渲染。要深入了解Vue模板编译,我们需要从编译过程、作用、特点等方面进行详细解析。 1. Vue模板编译的作用 Vue模板编译的主要作用是将Vue模板字符串…

021—pandas 书单整理将同一种书整理在一起

前言 在办公自动化场景下,最常见的需求就是信息的整理,pandas 最擅长复杂数据逻辑的处理,能够让整理工作更加高效,同时不容易出错。今天的案例是将一个平铺的书单按品类进行整理,合并为一行。 需求: 将书按书名进行合…

【python高级编程教程】笔记(python教程、python进阶)第三节:(1)多态与鸭子类型(Polymorphism and Duck Typing)

参考文章1:【比刷剧还爽】清华大佬耗时128小时讲完的Python高级教程!全套200集!学不会退出IT界! 参考文章2:清华教授大力打造的Python高级核心技术!整整100集,强烈建议学习(Python3…

基于springboot的母婴商城

摘 要 现代经济快节奏发展以及不断完善升级的信息化技术,让传统数据信息的管理升级为软件存储,归纳,集中处理数据信息的管理方式。本母婴商城系统就是在这样的大环境下诞生,其可以帮助管理者在短时间内处理完毕庞大的数据信息&am…

Windows10安装Docker

使用 PowerShell 启用 Hyper-V 以管理员身份打开 PowerShell 控制台。 运行以下命令: PowerShell复制 Enable-WindowsOptionalFeature -Online -FeatureName Microsoft-Hyper-V -All如果无法找到此命令,请确保你以管理员身份运行 PowerShell。 安装…

Claude3深夜震撼发布!模型特点分析,附使用教程

Claude3深夜震撼发布!模型特点分析,附使用教程 引言 最新发布的Claude3引起了广泛关注,这次发布一举推出了三个不同类型的模型,分别是Claude 3 Haiku、Claude 3 Sonnet和Claude 3 Opus。每个模型都具有独特的特点和能力&#xff…

Objective-C blocks 概要

1.block的使用 1.1什么是block? Blocks是C语言的扩充功能:带有自动变量(局部变量)的匿名函数。 “带有自动变量”在Blocks中表现为“截取自动变量" “匿名函数”就是“不带名称的函数” 块,封装了函数调用及调用…

代码随想录算法训练营Day38 || leetCode 7509. 斐波那契数 || 70. 爬楼梯 || 746. 使用最小花费爬楼梯

动态规划和我们数电中学习的时序电路类似,某一时刻的状态不仅与当前时刻的输入有关,还与之前的状态有关,所以推导过程中我们需要模拟题目中的情况,来找到每一时刻状态间的关系。 做题思路如下 509. 斐波那契数 此题简单 状态方程…

【QT】创建第一个QT程序

下面的前7个可以先不看,直接从8开始看 1. 创建Qt程序 一个Qt程序的组成部分:应用程序类,窗口类应用程序类个数:有且只有一个QApplication a;如何查看类对应的模块:光标移动到类上,F1qmake模块的名字 2. …

Java开发从零开始,mysql面试题索引

前言 “金九银十”的秋招热潮已经开始了,经过7月8月这两个月的提前批,终于成功拿下了一些大厂的offer。小编经过这么多次的面试,这两天整理了一份面试清单分享给大家,希望能给大家一点帮助(java方向),觉得有帮助的同学…

3分钟带你搞定电流采样电阻选型

大家好,我是砖一。 一,电流采样电阻的介绍 电流检测电路常用于高压短路保护、电机控制、DC/DC换流器、系统功耗管理、二次电池的电流管理、蓄电池管理等电流检测等场景。 比如,对于电机来说,电流检测电路是为了检测电流功能有比…

ssm基于javaEE+springboot校园闲置二手物品拍卖交易平台_ngad7

为提升浏览用户观感及使用体验,本系统要具有易用性和美观性。通过页面的简单提示就可完成操作,校园闲置物品交易平台展示界面应该清楚简洁,使用户通过美观的前台页面能快速定位想要浏览的校园闲置物品交易平台信息。后台界面也应简约&#xf…

领腾讯云红包,可抵扣云服务器订单金额

在2024年腾讯云新春采购节优惠活动上,可以领取新年惊喜红包,打开活动链接 https://curl.qcloud.com/oRMoSucP 会自动弹出红包领取窗口,如下图: 腾讯云2024新春采购节红包领取 如上图所示,点击“领”红包,每…

5G网络深度覆盖提升感知优化案例

随着5G业务的发展,用户感知尤为重要,随着人们的生活水平不断提高,对网络使用的要求也越来越高,用户感知更加重要,数据业务已超越语音业务成为流量和收入的主体,信号质量的决定作用更明显。5G TDD的频谱大带…

23蓝桥杯ACM培训-栈

前言&#xff1a; 回校第二天训练&#xff0c;今天的题目主要与stack有关。 正文&#xff1a; Problem:A 栈-程序员输入问题&#xff1a; #include<bits/stdc.h> using namespace std; int main(){stack<char> s1;stack<char> s2;string str;getline(cin,…

模仿Gitee实现站外链接跳转时进行确认

概述 如Gitee等网站&#xff0c;在有外部链接的时候如果不是同域则会出现一个确认页面。本文就带你看看这个功能应该如何实现。 效果 实现 1. 实现思路 将打开链接作为参数传递给一个中间页面&#xff0c;在页面加载的时候判断链接的域名和当前网站是否同域&#xff0c;同域…

贪心 Leetcode 763 划分字母区间

划分字母区间 Leetcode 763 学习记录自代码随想录 给你一个字符串 s 。我们要把这个字符串划分为尽可能多的片段&#xff0c;同一字母最多出现在一个片段中。 注意&#xff0c;划分结果需要满足&#xff1a;将所有划分结果按顺序连接&#xff0c;得到的字符串仍然是 s 。 返…

机器学习-面经(part7、无监督学习)

机器学习面经系列的其他部分如下所示&#xff1a; 机器学习-面经&#xff08;part1&#xff09; 机器学习-面经(part2)-交叉验证、超参数优化、评价指标等内容 机器学习-面经(part3)-正则化、特征工程面试问题与解答合集机器学习-面经(part4)-决策树共5000字的面试问题与解答…

C++ spfa求最短路 (可以带负权边)SPFA算法

给定一个 n 个点 m 条边的有向图&#xff0c;图中可能存在重边和自环&#xff0c; 边权可能为负数。 请你求出 1 号点到 n 号点的最短距离&#xff0c;如果无法从 1 号点走到 n 号点&#xff0c;则输出 impossible。 数据保证不存在负权回路。 输入格式 第一行包含整数 n 和…

一个基于差异同步数据库结构的工具 - Skeema

本文是 GO 三方库推荐的第 5 篇&#xff0c;继续介绍数据库 schema 同步工具&#xff0c;我前面已经写了两篇这个主题的文章。系列查看&#xff1a;Golang 三方库。 今天&#xff0c;推荐是的一个基于差异实现数据库 schema 迁移的工具库 - skeema&#xff0c;同样由 Go 实现。…