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

本文主要是介绍Spring揭秘:ImportBeanDefinitionRegistrar应用场景及实现原理!,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

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全面解析

精彩视频

这篇关于Spring揭秘:ImportBeanDefinitionRegistrar应用场景及实现原理!的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/782740

相关文章

JVM 的类初始化机制

前言 当你在 Java 程序中new对象时,有没有考虑过 JVM 是如何把静态的字节码(byte code)转化为运行时对象的呢,这个问题看似简单,但清楚的同学相信也不会太多,这篇文章首先介绍 JVM 类初始化的机制,然后给出几个易出错的实例来分析,帮助大家更好理解这个知识点。 JVM 将字节码转化为运行时对象分为三个阶段,分别是:loading 、Linking、initialization

Spring Security 基于表达式的权限控制

前言 spring security 3.0已经可以使用spring el表达式来控制授权,允许在表达式中使用复杂的布尔逻辑来控制访问的权限。 常见的表达式 Spring Security可用表达式对象的基类是SecurityExpressionRoot。 表达式描述hasRole([role])用户拥有制定的角色时返回true (Spring security默认会带有ROLE_前缀),去

浅析Spring Security认证过程

类图 为了方便理解Spring Security认证流程,特意画了如下的类图,包含相关的核心认证类 概述 核心验证器 AuthenticationManager 该对象提供了认证方法的入口,接收一个Authentiaton对象作为参数; public interface AuthenticationManager {Authentication authenticate(Authenti

Spring Security--Architecture Overview

1 核心组件 这一节主要介绍一些在Spring Security中常见且核心的Java类,它们之间的依赖,构建起了整个框架。想要理解整个架构,最起码得对这些类眼熟。 1.1 SecurityContextHolder SecurityContextHolder用于存储安全上下文(security context)的信息。当前操作的用户是谁,该用户是否已经被认证,他拥有哪些角色权限…这些都被保

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

Java架构师知识体认识

源码分析 常用设计模式 Proxy代理模式Factory工厂模式Singleton单例模式Delegate委派模式Strategy策略模式Prototype原型模式Template模板模式 Spring5 beans 接口实例化代理Bean操作 Context Ioc容器设计原理及高级特性Aop设计原理Factorybean与Beanfactory Transaction 声明式事物

大模型研发全揭秘:客服工单数据标注的完整攻略

在人工智能(AI)领域,数据标注是模型训练过程中至关重要的一步。无论你是新手还是有经验的从业者,掌握数据标注的技术细节和常见问题的解决方案都能为你的AI项目增添不少价值。在电信运营商的客服系统中,工单数据是客户问题和解决方案的重要记录。通过对这些工单数据进行有效标注,不仅能够帮助提升客服自动化系统的智能化水平,还能优化客户服务流程,提高客户满意度。本文将详细介绍如何在电信运营商客服工单的背景下进行

中文分词jieba库的使用与实景应用(一)

知识星球:https://articles.zsxq.com/id_fxvgc803qmr2.html 目录 一.定义: 精确模式(默认模式): 全模式: 搜索引擎模式: paddle 模式(基于深度学习的分词模式): 二 自定义词典 三.文本解析   调整词出现的频率 四. 关键词提取 A. 基于TF-IDF算法的关键词提取 B. 基于TextRank算法的关键词提取

水位雨量在线监测系统概述及应用介绍

在当今社会,随着科技的飞速发展,各种智能监测系统已成为保障公共安全、促进资源管理和环境保护的重要工具。其中,水位雨量在线监测系统作为自然灾害预警、水资源管理及水利工程运行的关键技术,其重要性不言而喻。 一、水位雨量在线监测系统的基本原理 水位雨量在线监测系统主要由数据采集单元、数据传输网络、数据处理中心及用户终端四大部分构成,形成了一个完整的闭环系统。 数据采集单元:这是系统的“眼睛”,