当前位置: 首页 > news >正文

SpringBoot系列-- @Enable 模块驱动

@Enable 模块驱动

@Enable 模块驱动是以 @Enable 为前缀的注解驱动编程模型。所谓 “模块” 是指具备相同领域的功能组件集合,组合所形成一个独立的单元。比如 WebMVC 模块、AspectJ 代理模块、Caching (缓存)模块、JMX (Java 管理扩展)模块、Async (异步处理)模块等。

举例说明
@EnableWebMvc
@EnableTransactionManagement
@EnableCaching
@EnableMBeanExport
@EnableAsync
@Enable 模块驱动编程模式

定义驱动注解:@EnableXXX
导入注解:@lmport 具体实现

实现方式两类

1. 基于注解方式实现

基于 Configuration Class类。在Enable 注解配置类中引入@Import 注解,导入配置类@Configuration 配置,实现配置类bean的初始化

  • 实现举例

实现配置Configration 配置类

@Configuration
public class HelloWorldConfigration {<!-- -->@Beanpublic String helloWorld() {<!-- -->return "Hello,World";}
}

自定义启动注解

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Import(HelloWorldConfigration.class)
public @interface EnableHelloWorld {<!-- -->
}

在启动类中启用使用模块

@EnableHelloWorld
@Configuration
public class EnableHelloworldBootstrap {<!-- -->public static void main(String[] args) {<!-- -->AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext();context.register(EnableHelloworldBootstrap.class);context.refresh();String helloworld = context.getBean("helloWorld", String.class);System.out.println("-------"+helloworld);context.close();}
}

2. 基于接口

分别获取注解信息,根据注解获取接入类型,根据接入类型选择不同的接入类型

举例
Access为接入类型的接口,下文的RPC接入和REST接入基于这个实现,定义两个接口,一个为启动,一个停止,内部嵌套一个枚举用于标识是哪一种接入

public interface Access {/*** 初始化配置*/void start();/*** 销毁配置*/void stop();enum Type{REST,RPC}
}

定义RPC和REST的实现

public class RestAccess implements Access{@Overridepublic void start() {System.out.println("rest接入配置");}@Overridepublic void stop() {System.out.println("rest接入销毁配置");}
}
public class RpcAccess implements Access{@Overridepublic void start() {System.out.println("rpc接入配置");}@Overridepublic void stop() {System.out.println("rpc接入销毁配置");}
}

自定义注解EnableAccess

@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
@Import(AccessImportSelector.class)
public @interface EnableAccess {/*** 接入类型* @return*/Access.Type type();
}

1. 基于 ImportSelector 接口实现

定义AccessImportSelector实现ImportSelector,分别获取注解信息,根据注解获取接入类型,根据接入类型选择不同的接入类型;使用spring的AnnotationMetadata 获取注解的元信息,通过注解的值判断需要实例化的bean对象;

public class AccessImportSelector implements ImportSelector{@Overridepublic String[] selectImports(AnnotationMetadata annotationMetadata) {//读取EnableAccess中所有的属性方法Map<String, Object> annotationAttributes = annotationMetadata.getAnnotationAttributes(EnableAccess.class.getName());//获取属性为type的属性方法Access.Type type = (Access.Type )annotationAttributes.get("type");//导入的类名称数组String [] importClassName = new String[0];switch (type){case RPC://设置为RPC,返回RpcAccess组件importClassName = new String[]{RpcAccess.class.getName()};break;case REST://设置为REST,返回RestAccess组件importClassName = new String[]{RestAccess.class.getName()};}return importClassName;}
}

使用

在primarySource也就是这里的DemoApplication上使用注解EnableAccess,选择接入方式,就会初始化不通的接入组件

@SpringBootApplication
@EnableAccess(type=Access.Type.REST)
public class DemoApplication {public static void main(String[] args) {ConfigurableApplicationContext context = SpringApplication.run(DemoApplication.class, args);Access access = context.getBean(Access.class);access.start();access.stop();}}

2. 基于 ImportBeanDefinitionRegistrar 接口实现

这里其它步骤一样,主要区别是注解里面Import的类变了,这里是基于基于ImportBeanDefinitionRegistrar实现注解驱动实现;

接口的使用很简单,使用@Import注解导入这个类即可

自定义ImportBeanDefinitionRegistrar

public class AccessImportBeanDefinitionRegistrar implements ImportBeanDefinitionRegistrar {@Overridepublic void registerBeanDefinitions(AnnotationMetadata annotationMetadata, BeanDefinitionRegistry beanDefinitionRegistry) {ImportSelector importSelector = new AccessImportSelector();//筛选class名称集合String[] selectedClassNames = importSelector.selectImports(annotationMetadata);Stream.of(selectedClassNames).map(BeanDefinitionBuilder::genericBeanDefinition).map(BeanDefinitionBuilder::getBeanDefinition).forEach(beanDefinition ->{//注册beanDefinition到beanDefinitionRegistryBeanDefinitionReaderUtils.registerWithGeneratedName(beanDefinition,beanDefinitionRegistry);});}
}

EnableAccess注解变更
这里import导入了AccessImportBeanDefinitionRegistrar

@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
@Import(AccessImportBeanDefinitionRegistrar.class)
public @interface EnableAccess {/*** 接入类型* @return*/Access.Type type();
}
  • 实现RPC接入type=Access.Type.RPC
@SpringBootApplication
@EnableAccess(type=Access.Type.RPC)
public class DemoApplication {public static void main(String[] args) {ConfigurableApplicationContext context = SpringApplication.run(DemoApplication.class, args);Access access = context.getBean(Access.class);access.start();access.stop();}}
  • REST接入type=Access.Type.REST
@SpringBootApplication
@EnableAccess(type=Access.Type.REST)
public class DemoApplication {public static void main(String[] args) {ConfigurableApplicationContext context = SpringApplication.run(DemoApplication.class, args);Access access = context.getBean(Access.class);access.start();access.stop();}}

importBeanDefinitionRegistrar接口是也是spring的扩展点之一,它可以支持我们自己写的代码封装成BeanDefinition对象;实现此接口的类会回调postProcessBeanDefinitionRegistry方法,注册到spring容器中。把bean注入到spring容器不止有 @Service @Component等注解方式;

public class ImportBeanDefinitionRegistrarTest implements ImportBeanDefinitionRegistrar{@Overridepublic void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {BeanDefinition beanDefinition = new GenericBeanDefinition();beanDefinition.setBeanClassName(TestBean.class.getName());MutablePropertyValues values = beanDefinition.getPropertyValues();values.addPropertyValue("id", 1);values.addPropertyValue("name", "ZhangSan");//这里注册beanregistry.registerBeanDefinition("testBean", beanDefinition );	}
}

基于注解方式和基于接口方式驱动模块的区别

基于接口方式明显的可以实现条件的配置,只需要启动类中通过枚举选择实例化的类型,缺点是启动的时候需要在代码中硬编码,对于开发有优势。

基于注解方式实现这种方式,可以基于spring的条件化配置@condition注解实现条件化的配置,实现条件配置的bean 实例化

http://www.lryc.cn/news/18491.html

相关文章:

  • PHP程序员适合创业吗?
  • 2023年CDGA考试-第12章-元数据(含答案)
  • 数据结构之顺序表篇
  • ZBC通证月内已翻倍,Nautilus Chain 上线前夕的“开门红”
  • 人工智能练习题:激活函数需要满足的条件、提高CNN的泛化能力、CNN输出特征图大小计算
  • KingbaseES Json 系列三:Json数据操作函数一
  • 《设计模式》单例模式
  • C/C++每日一练(20230224)
  • 基于YOLO的酸枣病虫害检测识别实践
  • WAF:ModSecurity on Nginx(15)
  • Qt 第3课、Qt 中的字符串类
  • Vulnhub靶场----6、DC-6
  • 华为OD机试真题Python实现【去重求和】真题+解题思路+代码(20222023)
  • lammps教程:Ovito选择特定晶粒的方法
  • DevEco Studio 3.1 Beta1版本发布——新增六大关键特性,开发更高效
  • 【蓝桥杯每日一题】二分算法
  • Spring Batch 高级篇-并行步骤
  • 对spring的@Cacheable缓存理解
  • 力扣-市场分析
  • 【2357. 使数组中所有元素都等于零】
  • 什么品牌的游戏蓝牙耳机比较好?玩游戏延迟低的蓝牙耳机推荐
  • day 33 状态压缩dp
  • 扬帆优配|超3600股飘绿,人民币贬值近300点!外资净卖近38亿
  • 【编程基础之Python】6、Python基础知识
  • selenium基本操作
  • 思科设备命令讲解(超基础二)
  • HTML基础(3)
  • 鸿蒙3.0 APP混合开发闪退问题笔记
  • 批量操作文件功能-课后程序(JAVA基础案例教程-黑马程序员编著-第七章-课后作业)
  • Hadoop3.3.1完全分布式部署