Spring條件注解沒生效該如何解決
從 Spring4.0 開始,Spring 提供了一個(gè)更加細(xì)粒度的條件注解: ConfigurationCondition。從名字上就可以看出來這個(gè)是搭配 @Configuration 注解一起使用的,ConfigurationCondition 提供了一種更加細(xì)粒度的條件匹配,可以在配置或者 Bean 注冊(cè)的時(shí)候去評(píng)估條件注解是否滿足。
也就是說,當(dāng)一個(gè)類上存在條件注解的時(shí)候,我們可以有兩個(gè)評(píng)估條件注解是否滿足的時(shí)機(jī):
- 在配置的時(shí)候去評(píng)估。
- 在 Bean 注冊(cè)的時(shí)候評(píng)估。
在配置的時(shí)候評(píng)估,可能會(huì)導(dǎo)致當(dāng)前類都不會(huì)被加載,在 Bean 注冊(cè)的時(shí)候再去評(píng)估,意味著當(dāng)前類就會(huì)被加載。
1. ConfigurationCondition
我們先來看下這個(gè)類的定義:
public interface ConfigurationCondition extends Condition { ConfigurationPhase getConfigurationPhase(); enum ConfigurationPhase { PARSE_CONFIGURATION, REGISTER_BEAN } }
大家看到,這里其實(shí)就是定義了兩個(gè)枚舉值,然后提供了一個(gè)方法返回枚舉值。
- PARSE_CONFIGURATION:這個(gè)表示 Condition 條件應(yīng)該在解析 @Configuration 類時(shí)進(jìn)行評(píng)估,如果評(píng)估不通過,則不會(huì)將 @Configuration 添加到容器中。
- REGISTER_BEAN:這個(gè)表示添加常規(guī) Bean 的時(shí)候去評(píng)估 Condition 條件(常規(guī) Bean 就是指非配置類,例如添加搭配 @Bean 注解使用的條件注解),這個(gè)條件不會(huì)阻止注冊(cè) @Configuration 類到容器中。
其實(shí)道理很好懂,就是加載配置類的時(shí)候就根據(jù)條件注解判斷要不要加載配置類,還是等到注冊(cè) Bean 的時(shí)候再去看條件注解是否滿足條件。
2. 案例分析
松哥通過一個(gè)簡(jiǎn)單案例來和小伙伴們演示一下。
假設(shè)我現(xiàn)在有如下條件:
public class MyCondition implements ConfigurationCondition { @Override public ConfigurationPhase getConfigurationPhase() { return ConfigurationPhase.PARSE_CONFIGURATION; } @Override public boolean matches(ConditionContext context, AnnotatedTypeMetadata metadata) { return context.getBeanFactory().containsBean("a"); } }
這個(gè)條件我沒有直接實(shí)現(xiàn) Condition 接口,而是實(shí)現(xiàn)類 ConfigurationCondition 接口,在這個(gè)接口中,getConfigurationPhase 方法返回了 PARSE_CONFIGURATION,表示在加載配置類的時(shí)候就去評(píng)估條件是否滿足,matches 方法則是去判斷容器中是否存在一個(gè)名為 a 的 Bean。
現(xiàn)在我有兩個(gè)配置類,分別是 A 和 B,如下:
@Configuration public class A { } @Configuration @Conditional(MyCondition.class) public class B { }
A 配置類正常加載,B 配置類有一個(gè)加載條件,就是得 A 存在,B 才會(huì)加載。
現(xiàn)在,在容器中加載 B 和 A 兩個(gè)配置,如下:
AnnotationConfigApplicationContext ctx = new AnnotationConfigApplicationContext(); ctx.register(B.class,A.class); ctx.refresh(); String[] beanDefinitionNames = ctx.getBeanDefinitionNames(); for (String beanDefinitionName : beanDefinitionNames) { System.out.println(beanDefinitionName); }
大家注意,加載的時(shí)候,我先加載了 B,后加載了 A,這點(diǎn)很重要,加載 B 的時(shí)候,由于此時(shí)容器中還不存在一個(gè)名為 a 的 Bean,而我們的評(píng)估時(shí)機(jī)是在處理配置類的時(shí)候,因此就會(huì)導(dǎo)致 B 配置類不會(huì)被加載,最終打印出來的 BeanName 就沒有 b。
但是,如果我們將 MyCondition 中,條件注解的評(píng)估時(shí)機(jī)改為 ConfigurationPhase.REGISTER_BEAN
,那么就表示在系統(tǒng)啟動(dòng)的時(shí)候,并不會(huì)去評(píng)估條件注解是否滿足,而是會(huì)將 @Configuration 配置類進(jìn)行解析,此時(shí)啟動(dòng)系統(tǒng),就會(huì)發(fā)現(xiàn)最終打印出來的 beanName 里既有 a 又有 b。
3. 源碼分析
接下來我們?cè)賮韽脑创a的角度來分析一下上述行為。
在 Spring 中,提供了一個(gè)專門的內(nèi)部類 ConditionEvaluator 來處理要不要跳過條件注解,該類中有一個(gè)名為 shouldSkip 的方法,用來處理此事:
public boolean shouldSkip(AnnotatedTypeMetadata metadata) { return shouldSkip(metadata, null); } public boolean shouldSkip(@Nullable AnnotatedTypeMetadata metadata, @Nullable ConfigurationPhase phase) { if (metadata == null || !metadata.isAnnotated(Conditional.class.getName())) { return false; } if (phase == null) { if (metadata instanceof AnnotationMetadata annotationMetadata && ConfigurationClassUtils.isConfigurationCandidate(annotationMetadata)) { return shouldSkip(metadata, ConfigurationPhase.PARSE_CONFIGURATION); } return shouldSkip(metadata, ConfigurationPhase.REGISTER_BEAN); } List<Condition> conditions = new ArrayList<>(); for (String[] conditionClasses : getConditionClasses(metadata)) { for (String conditionClass : conditionClasses) { Condition condition = getCondition(conditionClass, this.context.getClassLoader()); conditions.add(condition); } } AnnotationAwareOrderComparator.sort(conditions); for (Condition condition : conditions) { ConfigurationPhase requiredPhase = null; if (condition instanceof ConfigurationCondition configurationCondition) { requiredPhase = configurationCondition.getConfigurationPhase(); } if ((requiredPhase == null || requiredPhase == phase) && !condition.matches(this.context, metadata)) { return true; } } return false; }
第一個(gè)方法不用多說,我們來看第二個(gè)重載方法,重載方法多了一個(gè)參數(shù) ConfigurationPhase,這個(gè)就表示配置的階段,也就是條件注解生效的階段。
首先會(huì)去判斷當(dāng)前注解是否是一個(gè)條件注解,如果不是條件注意,那么就不能跳過,要繼續(xù)后面的解析(繼續(xù)后面的解析時(shí) Bean 將會(huì)被注冊(cè)),如果是條件注解,則繼續(xù)后面的判斷。繼續(xù)判斷,如果沒有傳遞 phase 進(jìn)來,說明沒有指定應(yīng)該在哪個(gè)階段去評(píng)估條件注解,那么這個(gè)時(shí)候就去判斷,如果當(dāng)前注解是一個(gè)配置類上的注解,那么就設(shè)置 phase 為 PARSE_CONFIGURATION,然后繼續(xù)調(diào)用 shouldSkip 方法,否則就設(shè)置 phase 為 REGISTER_BEAN 然后繼續(xù)調(diào)用 shouldSkip 方法。
那么什么樣的情況會(huì)被認(rèn)為是一個(gè)配置類上的注解呢?如果當(dāng)前類上添加的注解時(shí) @Component、@ComponentScan、@Import、@ImportResource 以及這四種注解衍生出來的注解,亦或者當(dāng)前類中有 @Bean 注解標(biāo)記的方法,那么當(dāng)前類就是一個(gè)配置類,就會(huì)設(shè)置 phase 為 PARSE_CONFIGURATION。
第二次進(jìn)入 shouldSkip 方法的時(shí)候,就已經(jīng)有明確的 phase 了。這次進(jìn)來后,把所有的條件注解的條件收集起來,存入到 conditions 集合中,然后再對(duì)該集合進(jìn)行排序。然后遍歷該集合。遍歷的時(shí)候就去判斷這個(gè)條件注解是不是 ConfigurationCondition 類型的,如果是,則提取出來其中的 phase 為 requiredPhase,這個(gè)就表示這個(gè)條件注意希望自己被處理的階段,接下來去判斷,如果 requiredPhase 為空,說明條件并未指定自己的執(zhí)行時(shí)間,那么就執(zhí)行 matches 方法進(jìn)行條件評(píng)估;如果 requiredPhase 不為空,并且和傳入的 phase 相等,那么也是當(dāng)前評(píng)估。其實(shí)這個(gè)判斷核心邏輯就是以參數(shù)傳入進(jìn)來的 phase 為準(zhǔn),要么條件沒有設(shè)置評(píng)估時(shí)機(jī),要么設(shè)置了,但是得和參數(shù)傳進(jìn)來的 phase 一致,只有滿足這兩個(gè)條件,才會(huì)當(dāng)場(chǎng)進(jìn)行評(píng)估。
這就是系統(tǒng)條件注解的評(píng)估邏輯。
對(duì)于配置類來說,是在 AnnotatedBeanDefinitionReader#doRegisterBean
方法中調(diào)用評(píng)估邏輯的:
private <T> void doRegisterBean(Class<T> beanClass, @Nullable String name, @Nullable Class<? extends Annotation>[] qualifiers, @Nullable Supplier<T> supplier, @Nullable BeanDefinitionCustomizer[] customizers) { AnnotatedGenericBeanDefinition abd = new AnnotatedGenericBeanDefinition(beanClass); if (this.conditionEvaluator.shouldSkip(abd.getMetadata())) { return; } //... }
調(diào)用的時(shí)候并未明確指定 phase,所以會(huì)在進(jìn)入到 shouldSkip 方法后,自行分析是哪個(gè)階段評(píng)估條件注解。
對(duì)于 @Bean 注解標(biāo)記的類來說,是在 ConfigurationClassBeanDefinitionReader#loadBeanDefinitionsForBeanMethod
方法中調(diào)用評(píng)估邏輯的:
private void loadBeanDefinitionsForBeanMethod(BeanMethod beanMethod) { ConfigurationClass configClass = beanMethod.getConfigurationClass(); MethodMetadata metadata = beanMethod.getMetadata(); String methodName = metadata.getMethodName(); if (this.conditionEvaluator.shouldSkip(metadata, ConfigurationPhase.REGISTER_BEAN)) { configClass.skippedBeanMethods.add(methodName); return; } //... }
這個(gè)調(diào)用的時(shí)候,就傳入了 phase 了,直接指定了是在 Bean 初始化的時(shí)候評(píng)估。
好啦,這就是條件注解條件評(píng)估時(shí)機(jī)的兩種情況。在 Spring Boot 中定義的條件注解里,有不少都用到了 ConfigurationCondition,而不是傳統(tǒng)的 Condition,感興趣的小伙伴可以自行查看哦~
到此這篇關(guān)于Spring條件注解沒生效該如何解決的文章就介紹到這了,更多相關(guān)Spring條件注解內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
springboot 設(shè)置server.port不生效的原因及解決
這篇文章主要介紹了springboot 設(shè)置server.port不生效的原因及解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-08-08基于@AllArgsConstructor與@Value共用的問題解決
這篇文章主要介紹了基于@AllArgsConstructor與@Value共用的問題解決,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-09-09SpringBoot項(xiàng)目中feignClient使用方式
文章介紹了在Spring Boot項(xiàng)目中配置Feign客戶端攔截器的具體步驟,包括在application.yml中添加配置、在主類上啟用組件掃描、將攔截器加入到攔截器列表中以及在接口調(diào)用時(shí)的說明,總結(jié)指出這是個(gè)人經(jīng)驗(yàn)分享,希望對(duì)大家有所幫助2024-11-11Spring Boot中使用RabbitMQ 生產(chǎn)消息和消費(fèi)消息的實(shí)例代碼
本文介紹了在SpringBoot中如何使用RabbitMQ進(jìn)行消息的生產(chǎn)和消費(fèi),詳細(xì)闡述了RabbitMQ中交換機(jī)的作用和類型,包括直連交換機(jī)、主題交換機(jī)、扇出交換機(jī)和頭交換機(jī),并解釋了各自的消息路由機(jī)制,感興趣的朋友一起看看吧2024-10-10Java多線程中的wait、notify和park、unpark的使用詳解
這篇文章主要介紹了Java多線程中的wait、notify和park、unpark的使用詳解,它們都是線程之間進(jìn)行協(xié)作的手段,都屬于 Object 對(duì)象的方法,必須獲得此對(duì)象的鎖,才能調(diào)用這幾個(gè)方法,需要的朋友可以參考下2023-12-12