Springboot基于enable模塊驅(qū)動的實現(xiàn)
enable作為模塊驅(qū)動在Spring Farmework、Spring Boot、Spring Cloud使用,都是通過注解的形式以@enable作為前綴,一些常用注解如
| 框架 | 注解 | 模塊 |
|---|---|---|
| Spring Framework | @EnableWebMvc | Web MVC模塊 |
| Spring Framework | @EnableTransactionmanagement | Web MVC模塊 |
| Spring Framework | @EnableCacheing | Cacheing模塊 |
| Spring Framework | @EnableMBeanExport | JMX模塊 |
| Spring Framework | @EnableWebFlux | Web Flux模塊 |
| Spring Framework | @EnableAspectJAutoProxy | AspectJ模塊 |
| Spring Boot | @EnableAutoConfiguration | 自動裝配模塊 |
| Spring Boot | @EnableWebManagementContext | Actuator模塊 |
| Spring Boot | @EnableConfigurationProperties | 配置屬性綁定模塊 |
| Spring Boot | @EnableOauth2Sso | OAuth2單獨登錄模塊 |
| Spring Cloud | @EnableEurekaServer | eureka服務(wù)模塊 |
| Spring Cloud | @EnableConfigServer | 配置服務(wù)器模塊 |
| Spring Cloud | @EnableFeignClients | Feign客戶端模塊 |
| Spring Cloud | @EnableZuulProxy | 服務(wù)網(wǎng)關(guān)zuul模塊 |
| Spring Cloud | @EnableCircuitBreaker | 服務(wù)熔斷模塊 |
如何自定義enable開發(fā)?
基于ImportSelector實現(xiàn)注解驅(qū)動
自定義接入類型
Access為接入類型的接口,下文的RPC接入和REST接入基于這個實現(xiàn),定義兩個接口,一個為啟動,一個停止,內(nèi)部嵌套一個枚舉用于標(biāo)識是哪一種接入
public interface Access {
/**
* 初始化配置
*/
void start();
/**
* 銷毀配置
*/
void stop();
enum Type{
REST,
RPC
}
}
定義RPC和REST的實現(xiàn)
REST實現(xiàn),只是簡單的打印方法
public class RestAccess implements Access{
@Override
public void start() {
System.out.println("rest接入配置");
}
@Override
public void stop() {
System.out.println("rest接入銷毀配置");
}
}
RPC實現(xiàn)
public class RpcAccess implements Access{
@Override
public void start() {
System.out.println("rpc接入配置");
}
@Override
public void stop() {
System.out.println("rpc接入銷毀配置");
}
}
自定義注解EnableAccess
接入類型為RPC或者REST,AccessImportSelector在下一步驟實現(xiàn)
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
@Import(AccessImportSelector.class)
public @interface EnableAccess {
/**
* 接入類型
* @return
*/
Access.Type type();
}
實現(xiàn)ImportSelector
定義AccessImportSelector實現(xiàn)ImportSelector,分別獲取注解信息,根據(jù)注解獲取接入類型,根據(jù)接入類型選擇不同的接入類型
public class AccessImportSelector implements ImportSelector{
@Override
public String[] selectImports(AnnotationMetadata annotationMetadata) {
//讀取EnableAccess中所有的屬性方法
Map<String, Object> annotationAttributes = annotationMetadata.getAnnotationAttributes(EnableAccess.class.getName());
//獲取屬性為type的屬性方法
Access.Type type = (Access.Type )annotationAttributes.get("type");
//導(dǎo)入的類名稱數(shù)組
String [] importClassName = new String[0];
switch (type){
case RPC:
//設(shè)置為RPC,返回RpcAccess組件
importClassName = new String[]{RpcAccess.class.getName()};
break;
case REST:
//設(shè)置為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();
}
}
基于ImportBeanDefinitionRegistrar實現(xiàn)注解驅(qū)動
這里其它步驟一樣,主要區(qū)別是注解里面Import的類變了,這里是基于基于ImportBeanDefinitionRegistrar實現(xiàn)注解驅(qū)動實現(xiàn)
自定義ImportBeanDefinitionRegistrar
public class AccessImportBeanDefinitionRegistrar implements ImportBeanDefinitionRegistrar {
@Override
public 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到beanDefinitionRegistry
BeanDefinitionReaderUtils.registerWithGeneratedName(beanDefinition,beanDefinitionRegistry);
});
}
}
EnableAccess注解變更
這里import導(dǎo)入了AccessImportBeanDefinitionRegistrar
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
@Import(AccessImportBeanDefinitionRegistrar.class)
public @interface EnableAccess {
/**
* 接入類型
* @return
*/
Access.Type type();
}
實現(xiàn)
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();
}
}
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Java?數(shù)據(jù)結(jié)構(gòu)與算法系列精講之?dāng)?shù)組
數(shù)組是有序的元素序列,若將有限個類型相同的變量的集合命名,那么這個名稱為數(shù)組名。組成數(shù)組的各個變量稱為數(shù)組的分量,也稱為數(shù)組的元素,有時也稱為下標(biāo)變量。數(shù)組是在程序設(shè)計中,為了處理方便, 把具有相同類型的若干元素按有序的形式組織起來的一種形式2022-02-02
深度解析Java中的國際化底層類ResourceBundle
做項目應(yīng)該都會實現(xiàn)國際化,那么大家知道Java底層是如何實現(xiàn)國際化的嗎?這篇文章就來和大家深度解析一下Java中的國際化底層類ResourceBundle,希望對大家有所幫助2023-03-03
idea使用帶provide修飾依賴導(dǎo)致ClassNotFound
程序打包到Linux上運行時,若Linux上也有這些依賴,為了在Linux上運行時避免依賴沖突,可以使用provide修飾,本文主要介紹了idea使用帶provide修飾依賴導(dǎo)致ClassNotFound,下面就來介紹一下解決方法,感興趣的可以了解一下2024-01-01
spring通過filter,Interceptor統(tǒng)一處理ResponseBody的返回值操作
這篇文章主要介紹了spring通過filter,Interceptor統(tǒng)一處理ResponseBody的返回值操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-09-09
關(guān)于maven pom文件配置加載lib下的jar包
在項目中引用放置在lib文件夾中的第三方j(luò)ar包需要通過POM文件進(jìn)行特定配置,雖然將依賴放置在公司的Maven私服是更好的做法,但如果遇到部署問題,可以利用maven-jar-plugin進(jìn)行配置,Spring Boot項目可以通過特定設(shè)置來實現(xiàn)2024-09-09

