在spring框架中,我们可以看到有许多的@Enablexxx注解,如spring的Cachine模块@EnableCaching,在springboot中有自动装配模块@EnableAutoConfiguration、OAuth2单点登录@EnableOAuth2Sso,在springcloud中还有Eureka服务器模块@EnableEurekaServer、Feign客户端模块@EnableFeignClients等@Enable注解。@Enablexx注解能够简化装配步骤,实现按需装配,同时屏蔽组件装配细节,不过要使用@Enable模块也必须手动触发,加注解在某个配置的bean上。
@Target(ElementType.TYPE)@Retention(RetentionPolicy.RUNTIME)@Documentedpublic @interface Import { /** * {@link Configuration @Configuration}, {@link ImportSelector},
* {@link ImportBeanDefinitionRegistrar}, or regular component classes to import.
*/
Class<?>[] value();}spring-boot-enable代码目录结构.png
@Datapublic class Student { private String name; private Integer age;}@Datapublic class Teacher { private String name; private Integer age;}@Configuration@Slf4jpublic class HelloWorldConfiguration { @Bean public Student student() { log.info("初始化student"); return new Student(); }}@Target(ElementType.TYPE)@Retention(RetentionPolicy.RUNTIME)@Documented@Import(HelloWorldConfiguration.class)public @interface EnableHelloWorld {}启动项目,我们会发现Student
bean并没有被springboot扫描并装载,然后在引导类SpringBootEnableApplication上面加注解@EnableHelloWorld,再次启动项目,通过日志可以看到Student
bean已经被初始化了。
@SpringBootApplication@EnableHelloWorldpublic class SpringBootEnableApplication { public static void main(String[] args) { SpringApplication.run(SpringBootEnableApplication.class, args); }}方式一注解驱动,启动项目.png
使用接口编程方式实现@Enable模块,需要实现ImportSelector类或ImportBeanDefinitionRegistrar类,相对于方式一,导入xxxConfiguration类,实现ImportSelector类和实现ImportBeanDefinitionRegistrar类的方式弹性更大,可以动态地选择一个或者多个@Componet类进行导入,使用的是Spring注解元信息AnnotationMetadata作为方法参数。
示例代码通过一个接口,两个接口实现类,在@Enable模块中通过传入枚举值实现动态选择其中一个实现类注册为Spring
Bean供controller层来使用。
public enum ActionType { /** * 老师
*/
TEACHER,
/**
* 学生
*/
STUDENT
}@Target(ElementType.TYPE)@Retention(RetentionPolicy.RUNTIME)@Documented@Import(ServerImportSelector.class)public @interface EnableActionServer { /** * 设置动作类型,默认为老师上课
*/
ActionType serverType() default ActionType.TEACHER;
}public interface ActionServer { /** * 动作
*/
String action();
}@Slf4j@Servicepublic class StudentActionServerImpl implements ActionServer { @Bean public Student studentone() { log.info("初始化student bean"); return new Student(); } @Override public String action() { log.info("学生上课"); return "学生上课"; }}@Slf4j@Servicepublic class TeacherActionServerImpl implements ActionServer { @Bean public Teacher teacherone() { log.info("初始化teacher bean"); return new Teacher(); } @Override public String action() { log.info("老师上课"); return "老师上课"; }}@Slf4jpublic class ServerImportSelector implements ImportSelector { @Override public String[] selectImports(AnnotationMetadata importingClassMetadata) { //获取注解@EnableActionServer元信息 Map<String, Object> annotationAttributes = importingClassMetadata.getAnnotationAttributes(EnableActionServer.class.getName()); assert annotationAttributes != null; //通过获取的注解@EnableActionServer元信息,取定义好的方法属性,并把对象Object强制转换为ActionType枚举类型 ActionType serverType = (ActionType) annotationAttributes.get("serverType"); log.info("serverType:{},getServerType(serverType):{}", serverType, getServerType(serverType)); //最后返回要选择的类名,注册为spring bean组件 return new String[]{getServerType(serverType)}; } //封装一层,通过枚举值返回不同的类名 public String getServerType(ActionType serverType) { Map<ActionType, String> serverClassMap; serverClassMap = new HashMap<>(2); serverClassMap.put(ActionType.TEACHER, TeacherActionServerImpl.class.getName()); serverClassMap.put(ActionType.STUDENT, StudentActionServerImpl.class.getName()); return serverClassMap.get(serverType); }}在controller层注入ActionServer,然后启动项目,可以发现报错了。报错是因为没有定义好Spring
Bean,题外话:当然可以通过@Bean的方式去注入某个实现类或者在实现类上加@ Server注解再选择其中之一的实现类去注入。
@RestController@RequestMapping("/api/actionServer")public class ActionServerController { @Autowired private ActionServer actionServer; @GetMapping("/actionType") public String actionType() { return actionServer.action(); }}注入ActionServer类报错.png
最后,我们也用实现ImportSelector接口类的方式去实现@Enable模块,那么在引导类上加上注解@EnableActionServer(serverType
= ActionType.STUDENT)即可,再次启动项目
@SpringBootApplication@EnableActionServer(serverType = ActionType.STUDENT)public class SpringBootEnableApplication { public static void main(String[] args) { SpringApplication.run(SpringBootEnableApplication.class, args); }}方式二启动项目.png
浏览器访问接口.png
ImportBeanDefinitionRegistrar相对于ImportSelector而言,编程复杂度更高,除了注解元信息AnnotationMetadata作为入参外,接口将定义Bean的注册交给开发人员。示例代码复用方式二的选择实现类逻辑,故直接通过new
ServerImportSelector()的方式来复用实现好的selectImports方法,入参即为注解元信息AnnotationMetadata,返回类名,最后完成注册返回类名的类为Spring
Bean。
public class ServerImportBeanDefinitionRegistrar implements ImportBeanDefinitionRegistrar { @Override public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) { //复用前面实现好的ImportSelector ImportSelector importSelector = new ServerImportSelector(); //筛选ClassNames集合 String[] selectClassNames = importSelector.selectImports(importingClassMetadata); Stream.of(selectClassNames) //转化为 BeanDefinitionBuilder 对象 .map(BeanDefinitionBuilder::genericBeanDefinition) //转化为 BeanDefinition 对象 .map(BeanDefinitionBuilder::getBeanDefinition) //注册 BeanDefinition 到 BeanDefinitionRegistry .forEach(beanDefinition -> BeanDefinitionReaderUtils.registerWithGeneratedName(beanDefinition, registry)); }}接着修改前面写好的注解@EnableActionServer注解的@Import导入类,修改为@Import(ServerImportBeanDefinitionRegistrar.class)
@Target(ElementType.TYPE)@Retention(RetentionPolicy.RUNTIME)@Documented//@Import(ServerImportSelector.class)@Import(ServerImportBeanDefinitionRegistrar.class)public @interface EnableActionServer { /** * 设置动作类型,默认为老师上课
*/
ActionType serverType() default ActionType.TEACHER;
}再次启动项目即可观察结果。
原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。
如有侵权,请联系 cloudcommunity@tencent.com 删除。
原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。
如有侵权,请联系 cloudcommunity@tencent.com 删除。