Spring中@Condition调用过程

本文基于
Spring 5.1.7.release
SpringBoot 2.1.5.release

Spring 中的 @Condition 能非常方便帮助我们条件化的注册 bean。这里我不做 @Conditional 相关注解的介绍。这里介绍下 Condition 的调用过程和一些注意事项。

今天在看事务自动配置类 TransactionAutoConfiguration 时发现静态内部类 EnableTransactionManagementConfiguration 的注册并不是我预期的样子, TransactionAutoConfiguration 的代码如下:

package org.springframework.boot.autoconfigure.transaction;
...
...
/**
 * {@link org.springframework.boot.autoconfigure.EnableAutoConfiguration
 * Auto-configuration} for Spring transaction.
 *
 * @author Stephane Nicoll
 * @since 1.3.0
 */
@Configuration
@ConditionalOnClass(PlatformTransactionManager.class)
@AutoConfigureAfter({ JtaAutoConfiguration.class, HibernateJpaAutoConfiguration.class,
        DataSourceTransactionManagerAutoConfiguration.class,
        Neo4jDataAutoConfiguration.class })
@EnableConfigurationProperties(TransactionProperties.class)
public class TransactionAutoConfiguration {

    @Bean
    @ConditionalOnMissingBean
    public TransactionManagerCustomizers platformTransactionManagerCustomizers(
            ObjectProvider<PlatformTransactionManagerCustomizer<?>> customizers) {
        return new TransactionManagerCustomizers(
                customizers.orderedStream().collect(Collectors.toList()));
    }

    @Configuration
    @ConditionalOnSingleCandidate(PlatformTransactionManager.class)
    public static class TransactionTemplateConfiguration {

        private final PlatformTransactionManager transactionManager;

        public TransactionTemplateConfiguration(
                PlatformTransactionManager transactionManager) {
            this.transactionManager = transactionManager;
        }

        @Bean
        @ConditionalOnMissingBean
        public TransactionTemplate transactionTemplate() {
            return new TransactionTemplate(this.transactionManager);
        }

    }

    @Configuration
    @ConditionalOnBean(PlatformTransactionManager.class)
    @ConditionalOnMissingBean(AbstractTransactionManagementConfiguration.class)
    public static class EnableTransactionManagementConfiguration {

        @Configuration
        @EnableTransactionManagement(proxyTargetClass = false)
        @ConditionalOnProperty(prefix = "spring.aop", name = "proxy-target-class",
                havingValue = "false", matchIfMissing = false)
        public static class JdkDynamicAutoProxyConfiguration {

        }

        @Configuration
        @EnableTransactionManagement(proxyTargetClass = true)
        @ConditionalOnProperty(prefix = "spring.aop", name = "proxy-target-class",
                havingValue = "true", matchIfMissing = true)
        public static class CglibAutoProxyConfiguration {

        }

    }

}

可以看到 EnableTransactionManagementConfiguration 使用了 :

  • @ConditionalOnBean(PlatformTransactionManager.class)
  • @ConditionalOnMissingBean(AbstractTransactionManagementConfiguration.class)

起初我认为如果 BeanFactory 中有 PlatformTransactionManager 且没有 AbstractTransactionManagementConfiguration 才会进入内部进行内部的初始化(也就是说 @EnableTransactionManagement 注解不会起作用,也就是说不会进入 TransactionManagementConfigurationSelector ),但是当我把 PlatformTransactionManager 的 bean 定义去除后,发现还是会进入到 TransactionManagementConfigurationSelector,这使我很疑惑。经过一系列 debug 后终于找到了原因。

Condition 原理

首先所有的 Configuration 都会被 ConfigurationClassPostProcessor#postProcessBeanDefinitionRegistry 处理,之后会被 ConfigurationClassParser#parse(Set<BeanDefinitionHolder> configCandidates) 解析。在解析时会调用 ConditionEvaluator#shouldSkip 判断是否需要把目标解析为 ConfigurationClass,解析时的判断源码如下:

1. ConfigurationClassParser#parse -> ConfigurationClassParser#processConfigurationClass
protected void processConfigurationClass(ConfigurationClass configClass) throws IOException {
        if (this.conditionEvaluator.shouldSkip(configClass.getMetadata(), ConfigurationPhase.PARSE_CONFIGURATION)) {
            return;
        }

        ConfigurationClass existingClass = this.configurationClasses.get(configClass);
        if (existingClass != null) {
            if (configClass.isImported()) {
                if (existingClass.isImported()) {
                    existingClass.mergeImportedBy(configClass);
                }
                // Otherwise ignore new imported config class; existing non-imported class overrides it.
                return;
            }
            else {
                // Explicit bean definition found, probably replacing an import.
                // Let's remove the old one and go with the new one.
                this.configurationClasses.remove(configClass);
                this.knownSuperclasses.values().removeIf(configClass::equals);
            }
        }

        // Recursively process the configuration class and its superclass hierarchy.
        SourceClass sourceClass = asSourceClass(configClass);
        do {
            sourceClass = doProcessConfigurationClass(configClass, sourceClass);
        }
        while (sourceClass != null);

        this.configurationClasses.put(configClass, configClass);
    }

shouldSkip 判断是否跳过 configClass 的解析,注意这里使用的是ConfigurationPhase.PARSE_CONFIGURATION,这也是为什么我之前的以为是错误的原因。
ConfigurationPhase.PARSE_CONFIGURATION 表示 Condition 只有在解析阶段才起作用,也就是说 Condition 不是 ConfigurationCondition 的实现或者阶段不匹配都不会跳过

2. shouldSkip
/**
  * false: 不跳过,true:跳过
  */
public boolean shouldSkip(@Nullable AnnotatedTypeMetadata metadata, @Nullable ConfigurationP
    // metadata == null 或 没有注解 @Conditional 相关的类,直接返回 false
    if (metadata == null || !metadata.isAnnotated(Conditional.class.getName())) {
        return false;
    }
    // 如果 phase  (阶段为空),判断类上是否有 @Configuration, @Component,@ComponentScan,@Import,@ImportResource 
    // 或方法上有 @Bean 注解,如果有就使用 ConfigurationPhase.PARSE_CONFIGURATION)
    if (phase == null) {
        if (metadata instanceof AnnotationMetadata &&
                ConfigurationClassUtils.isConfigurationCandidate((AnnotationMetadata) metada
            return shouldSkip(metadata, ConfigurationPhase.PARSE_CONFIGURATION);
        }
        return shouldSkip(metadata, ConfigurationPhase.REGISTER_BEAN);
    }
    List<Condition> conditions = new ArrayList<>();
     // 获取 @Conditional 注解对应得 Condtion 放入 conditions,这里使用了反射
    for (String[] conditionClasses : getConditionClasses(metadata)) {
        for (String conditionClass : conditionClasses) {
            Condition condition = getCondition(conditionClass, this.context.getClassLoader()
            conditions.add(condition);
        }
    }
    // 排序
    AnnotationAwareOrderComparator.sort(conditions);
    // 遍历 Condition 判断是否应该跳过
    for (Condition condition : conditions) {
        ConfigurationPhase requiredPhase = null;
        // 如果 Condition 实现了 ConfigurationCondition 则获取该 Condition 得阶段
        if (condition instanceof ConfigurationCondition) {
            requiredPhase = ((ConfigurationCondition) condition).getConfigurationPhase();
        }
        // 如果 (requiredPhase == null || 阶段匹配 ) && matches 返回 false 就返回 true
        // 如果 requiredPhase != null && 阶段不匹配 直接返回 false
        if ((requiredPhase == null || requiredPhase == phase) && !condition.matches(this.context, metadata)) {
            return true;
        }
    }
    return false;
}

看到这我们会看

  • @ConditionalOnBean(PlatformTransactionManager.class)
@Target({ ElementType.TYPE, ElementType.METHOD })
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Conditional(OnBeanCondition.class)
public @interface ConditionalOnBean {
  • @ConditionalOnMissingBean(AbstractTransactionManagementConfiguration.class)
@Target({ ElementType.TYPE, ElementType.METHOD })
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Conditional(OnBeanCondition.class)
public @interface ConditionalOnMissingBean {

可以看到这两个 @Conditonal 都是用得是 OnBeanCondition 来处理。翻看源码发现 OnBeanCondition 实现了 ConfigurationCondition,它的处理阶段是 ConfigurationPhase.REGISTER_BEAN(也就是注册阶段)。

看到这,相信大家对整个得 Condition 处理逻辑有了大概得了解,也明白为什么我起初得错误是错误的吧。

Conclusion

在使用 @Condtional 相关注解时需要注意对应的 Condition 起作用的阶段,防止使用错误。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,539评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,911评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,337评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,723评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,795评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,762评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,742评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,508评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,954评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,247评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,404评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,104评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,736评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,352评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,557评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,371评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,292评论 2 352

推荐阅读更多精彩内容

  • AA孟雪阅读 295评论 0 0
  • 合肥16度,深圳26度。 虽然离开了你所在的城市,心却天天挂念那里的天气,担心你的冷暖。已经回来两天了,除了天...
    晨定阅读 99评论 0 0
  • AI的价值地图,详细的阐明了开源技术平台的核心技术,创造开放技术平台到技术操作系统到应用解决方案到商业运营系统,直...
    Binner阅读 473评论 0 50
  • 前一段时间吧!因为有一个长长的暑假,所以一直在写文章,本来就是想打击打击我们家领导嚣张的气焰。记得我有写过这样一句...
    August_77e9阅读 373评论 0 9
  • 今天这篇文章,不是我的原创,是芷姑娘的文章,但看过有些感触,就和芷姑娘要了版权。文章如下—— 今天想和大家分享的一...
    喵二叔阅读 365评论 0 1