@Conditional注解 -【Spring底层原理】

r囧r小猫 2021-09-21 17:36 504阅读 0赞

案例已上传GitHub,欢迎star以鼓励:https://github.com/oneStarLR/spring-annotation

一、注解用法

@Conditional是Spring4新提供的注解,也是用来注册bean的,作用如下:

  • 按照一定的条件进行判断,满足条件的给容器注册bean
  • 从源码中我们可以看到,可以作用在类和方法上
  • 需要传入一个Class数组,并继承Condition接口
  1. // 可以作用在类上,也可以作用在方法上
  2. @Target({ElementType.TYPE, ElementType.METHOD})
  3. @Retention(RetentionPolicy.RUNTIME)
  4. @Documented
  5. public @interface Conditional {
  6. // 需要传入一个Class数组
  7. Class<? extends Condition>[] value();
  8. }
  9. // 继承Condition接口
  10. @FunctionalInterface
  11. public interface Condition {
  12. boolean matches(ConditionContext var1, AnnotatedTypeMetadata var2);
  13. }

在继承Condition接口中,我们可以获取上下文环境,从而进行判断,达到条件判断的作用

二、实例分析

通过实例来进行分析,以不同的操作系统为条件,通过实现Condition接口,并重写其matches方法来构造判断条件,通过idea配置来改变操作系统环境,将注入的bean进行打印来进行判断。

  1. // 启动类
  2. @Test
  3. public void TestMain(){
  4. AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext(AppConfig.class);
  5. String[] beanNames = applicationContext.getBeanDefinitionNames();
  6. for (String beanName : beanNames) {
  7. System.out.println(beanName);
  8. }
  9. }
  10. // User
  11. public class User {
  12. }
  13. // 配置类
  14. @Configuration
  15. public class AppConfig {
  16. @Bean
  17. public User user1(){
  18. return new User();
  19. }
  20. @Bean
  21. public User user2(){
  22. return new User();
  23. }
  24. }

上面的代码,通过启动测试类,会将user1和user2注入到容器,可以看到打印结果如下:

image-20210221204501107

现在需要根据操作系统来进行条件注入,Windows系统下注入user1,Linux系统下注入user2,则需要实现Condition接口,并重写其matches方法来构造判断条件

  • 实现Condition接口:Windows系统判断条件

    // Windows系统判断条件
    public class WindowsCondition implements Condition {

    1. /**
    2. * @description TODO
    3. * @author ONESTAR
    4. * @date 2021/2/10 10:56
    5. * @param conditionContext:判断条件,能使用的上下问环境
    6. * @param annotatedTypeMetadata:注释信息
    7. * @throws
    8. * @return boolean
    9. */
    10. public boolean matches(ConditionContext conditionContext, AnnotatedTypeMetadata annotatedTypeMetadata) {
    11. // 获取当前环境
    12. Environment environment = conditionContext.getEnvironment();
    13. // 判断是否是Windows系统
    14. String property = environment.getProperty("os.name");
    15. if (property.contains("Windows")){
    16. return true;
    17. }
    18. return false;
    19. }

    }

  • 实现Condition接口:Linux系统判断条件

    // Linux系统判断条件
    public class LinuxCondition implements Condition {

    1. /**
    2. * @description 判断操作系统是否是Linux系统
    3. * @author ONESTAR
    4. * @date 2021/2/10 10:56
    5. * @param conditionContext
    6. * @param annotatedTypeMetadata
    7. * @throws
    8. * @return boolean
    9. */
    10. public boolean matches(ConditionContext conditionContext, AnnotatedTypeMetadata annotatedTypeMetadata) {
    11. // 获取当前环境
    12. Environment environment = conditionContext.getEnvironment();
    13. // 判断是否是Linux系统
    14. String property = environment.getProperty("os.name");
    15. if (property.contains("linux")){
    16. return true;
    17. }
    18. return false;
    19. }

    }

  • 修改配置类,使用@Conditional注解进行条件注入,修改后如下

    @Configuration
    public class AppConfig {

    1. // 如果WindowsCondition的实现方法返回true,则注入这个bean
    2. @Conditional({WindowsCondition.class})
    3. @Bean
    4. public User user1(){
    5. return new User();
    6. }
    7. // 如果LinuxCondition的实现方法返回true,则注入这个bean
    8. @Conditional({LinuxCondition.class})
    9. @Bean
    10. public User user2(){
    11. return new User();
    12. }

    }

这时我们再来运行启动类,默认情况下是Windows系统,可以看到,只有user1注入进去了,user2并没有注入

image-20210221205808356

咱们通过idea配置来模拟改变运行环境:添加:-Dos.name=linux

image-20210221210032021

image-20210221205930027

改变运行环境后,咱们再来运行启动类,可以看到,此时注入的是user2:

image-20210221210152727

三、源码追踪

参考:https://www.jianshu.com/p/566f22bda03c

【1】ConditionEvaluatormatches方法

我们知道,spring通过实现Condition接口,并重写其matches方法来构造判断条件,可以从matches入手,查看源码,发现ConditionEvaluator中调用了matches这个方法

image-20210224101144219

  1. public boolean shouldSkip(@Nullable AnnotatedTypeMetadata metadata, @Nullable ConfigurationPhase phase) {
  2. // 检查注解中是否包含@Conditional类型的注解
  3. if (metadata != null && metadata.isAnnotated(Conditional.class.getName())) {
  4. // 判断当前bean是解析还是注册
  5. if (phase == null) {
  6. // bean的注解信息封装对象是AnnotationMetadata类型并且,类上有@Component,@ComponentScan,@Import,@ImportResource,则表示为解析类型
  7. return metadata instanceof AnnotationMetadata && ConfigurationClassUtils.isConfigurationCandidate((AnnotationMetadata)metadata) ? this.shouldSkip(metadata, ConfigurationPhase.PARSE_CONFIGURATION) : this.shouldSkip(metadata, ConfigurationPhase.REGISTER_BEAN);
  8. } else {
  9. List<Condition> conditions = new ArrayList();
  10. Iterator var4 = this.getConditionClasses(metadata).iterator();
  11. // 从bean的注解信息封装对象中获取所有的Conditional类型或者Conditional的派生注解
  12. while(var4.hasNext()) {
  13. String[] conditionClasses = (String[])var4.next();
  14. String[] var6 = conditionClasses;
  15. int var7 = conditionClasses.length;
  16. for(int var8 = 0; var8 < var7; ++var8) {
  17. String conditionClass = var6[var8];
  18. // 实例化Conditional中的条件判断类(Condition的子类)
  19. Condition condition = this.getCondition(conditionClass, this.context.getClassLoader());
  20. // 添加到条件集合中
  21. conditions.add(condition);
  22. }
  23. }
  24. // 根据Condition的优先级进行排序
  25. AnnotationAwareOrderComparator.sort(conditions);
  26. var4 = conditions.iterator();
  27. Condition condition;
  28. ConfigurationPhase requiredPhase;
  29. do {
  30. do {
  31. if (!var4.hasNext()) {
  32. return false;
  33. }
  34. condition = (Condition)var4.next();
  35. requiredPhase = null;
  36. // 如果是ConfigurationCondition类型的Condition
  37. if (condition instanceof ConfigurationCondition) {
  38. // 获取需要对bean进行的操作,是解析还是注册
  39. requiredPhase = ((ConfigurationCondition)condition).getConfigurationPhase();
  40. }
  41. //(如果requiredPhase==null或者指定的操作类型是目前阶段的操作类型)并且不符合设置的条件则跳过
  42. } while(requiredPhase != null && requiredPhase != phase);
  43. } while(condition.matches(this.context, metadata));
  44. return true;
  45. }
  46. } else {
  47. return false;
  48. }
  49. }

ConditionEvaluator这个类的作用是评估一个加了Conditional注解的类是否需要跳过。通过类上面的注解来判断。该方法作用就是判断当前bean处于解析还是注册

  • 如果处于解析阶段则跳过,如果处于注册阶段则不跳过。
  • 其中Conditionmatches方法就起到了判断的是否符合的作用,进而判断是否跳过当前bean。

【2】ConfigurationClassPostProcessorprocessConfigBeanDefinitions

还是通过查找ConditionEvaluator类的matches方法调用链的方式,发现最后都是在ConfigurationClassPostProcessorprocessConfigBeanDefinitions中进行调用的。一共有两个调用的位置,这里用调用的位置的代码进行展示

  1. public void processConfigBeanDefinitions(BeanDefinitionRegistry registry) {
  2. List<BeanDefinitionHolder> configCandidates = new ArrayList();
  3. // 获取registry中定义的所有的bean的name
  4. String[] candidateNames = registry.getBeanDefinitionNames();
  5. ......
  6. do {
  7. // 第一个会调用shouldSkip的位置,这里是解析能够直接获取的候选配置bean。可能是Component,ComponentScan,Import,ImportResource或者有Bean注解的bean
  8. StartupStep processConfig = this.applicationStartup.start("spring.context.config-classes.parse");
  9. parser.parse(candidates);
  10. parser.validate();
  11. // 获取上面封装已经解析过的配置bean的ConfigurationClass集合
  12. Set<ConfigurationClass> configClasses = new LinkedHashSet(parser.getConfigurationClasses());
  13. // 移除前面已经处理过的
  14. configClasses.removeAll(alreadyParsed);
  15. if (this.reader == null) {
  16. this.reader = new ConfigurationClassBeanDefinitionReader(registry, this.sourceExtractor, this.resourceLoader, this.environment, this.importBeanNameGenerator, parser.getImportRegistry());
  17. }
  18. //第二个会调用shouldSkip的位置,这里是加载configurationClasse中内部可能存在配置bean,比如方法上加了@Bean或者@Configuration标签的bean
  19. this.reader.loadBeanDefinitions(configClasses);
  20. alreadyParsed.addAll(configClasses);
  21. ......
  22. }
  23. }
  • 通过parse方法解析BeanDefinitionRegistry中能直接获取到的候选bean,并解析保存到ConfigurationClassParser类的保存解析过的配置类的集合configurationClasses
  • loadBeanDefinitions则是对上面解析的集合configurationClasses中的bean内部的进一步的处理,处理类内部定义的bean

【3】ConfigurationClassParserparse方法

  1. public void parse(Set<BeanDefinitionHolder> configCandidates) {
  2. Iterator var2 = configCandidates.iterator();
  3. while(var2.hasNext()) {
  4. BeanDefinitionHolder holder = (BeanDefinitionHolder)var2.next();
  5. BeanDefinition bd = holder.getBeanDefinition();
  6. try {
  7. if (bd instanceof AnnotatedBeanDefinition) {
  8. this.parse(((AnnotatedBeanDefinition)bd).getMetadata(), holder.getBeanName());
  9. } else if (bd instanceof AbstractBeanDefinition && ((AbstractBeanDefinition)bd).hasBeanClass()) {
  10. this.parse(((AbstractBeanDefinition)bd).getBeanClass(), holder.getBeanName());
  11. } else {
  12. this.parse(bd.getBeanClassName(), holder.getBeanName());
  13. }
  14. } catch (BeanDefinitionStoreException var6) {
  15. throw var6;
  16. } catch (Throwable var7) {
  17. throw new BeanDefinitionStoreException("Failed to parse configuration class [" + bd.getBeanClassName() + "]", var7);
  18. }
  19. }
  20. this.deferredImportSelectorHandler.process();
  21. }
  22. protected final void parse(@Nullable String className, String beanName) throws IOException {
  23. Assert.notNull(className, "No bean class name for configuration class bean definition");
  24. MetadataReader reader = this.metadataReaderFactory.getMetadataReader(className);
  25. this.processConfigurationClass(new ConfigurationClass(reader, beanName), DEFAULT_EXCLUSION_FILTER);
  26. }
  27. protected final void parse(Class<?> clazz, String beanName) throws IOException {
  28. this.processConfigurationClass(new ConfigurationClass(clazz, beanName), DEFAULT_EXCLUSION_FILTER);
  29. }
  30. protected final void parse(AnnotationMetadata metadata, String beanName) throws IOException {
  31. this.processConfigurationClass(new ConfigurationClass(metadata, beanName), DEFAULT_EXCLUSION_FILTER);
  32. }

ConfigurationClassParserparse方法中有三个分支,分别是对不同类型的BeanDefinition进行解析,这里进入AnnotatedBeanDefinition类型的。

【4】调用processConfigurationClass方法

进入到parse方法后在进入里面调用的processConfigurationClass方法,查看源码,这里就是对Conditional注解的作用了

  1. protected void processConfigurationClass(ConfigurationClass configClass, Predicate<String> filter) throws IOException {
  2. // 检查当前解析的配置bean是否包含Conditional注解,如果不包含则不需要跳过
  3. // 如果包含了则进行match方法得到匹配结果
  4. if (!this.conditionEvaluator.shouldSkip(configClass.getMetadata(), ConfigurationPhase.PARSE_CONFIGURATION)) {
  5. ConfigurationClass existingClass = (ConfigurationClass)this.configurationClasses.get(configClass);
  6. if (existingClass != null) {
  7. if (configClass.isImported()) {
  8. if (existingClass.isImported()) {
  9. existingClass.mergeImportedBy(configClass);
  10. }
  11. return;
  12. }
  13. this.configurationClasses.remove(configClass);
  14. this.knownSuperclasses.values().removeIf(configClass::equals);
  15. }
  16. ConfigurationClassParser.SourceClass sourceClass = this.asSourceClass(configClass, filter);
  17. do {
  18. sourceClass = this.doProcessConfigurationClass(configClass, sourceClass, filter);
  19. } while(sourceClass != null);
  20. this.configurationClasses.put(configClass, configClass);
  21. }
  22. }

这里就是对是否跳过bean解析的位置

  • 检查当前解析的配置bean是否包含Conditional注解,如果不包含则不需要跳过
  • 如果包含了则进行match方法得到匹配结果,如果是符合的并且设置的配置解析策略是解析阶段不需要调过

四、总结

@Conditional注解主要通过指定的Condition实现类实现matches方法来决定是否需要进行解析,总结如下:

  1. 通过实现Condition接口,并重写其matches方法来构造判断条件
  2. 通过ConditionEvaluatormatches方法判断当前bean处于解析还是注册,如果处于解析阶段则跳过,如果处于注册阶段则不跳过
  3. 调用processConfigurationClass方法判断当前解析的配置bean是否包含Conditional注解,如果不包含则不需要跳过,包含了则进行match方法得到匹配结果

发表评论

表情:
评论列表 (有 0 条评论,504人围观)

还没有评论,来说两句吧...

相关阅读