欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Spring條件注解沒生效該如何解決

 更新時(shí)間:2023年09月11日 14:54:53   作者:江南一點(diǎn)雨  
條件注解相信各位小伙伴都用過,Spring?中的多環(huán)境配置?profile?底層就是通過條件注解來實(shí)現(xiàn)的,下面小編就來為大家介紹一下當(dāng)Spring條件注解沒生效時(shí)該如何解決,感興趣的可以了解下

從 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)文章

  • 詳解Java中的泛型

    詳解Java中的泛型

    這篇文章主要介紹了Java中的泛型,當(dāng)我們不確定數(shù)據(jù)類型時(shí),我們可以暫時(shí)使用一個(gè)字母 T代替數(shù)據(jù)類型,例如寫一個(gè)方法,但是我們不知道它是傳遞的是什么數(shù)據(jù)類型,我們就可以使用泛型,到時(shí)候只要指明T是什么數(shù)據(jù)類型,就可以使用了,需要的朋友可以參考下
    2023-05-05
  • springboot 設(shè)置server.port不生效的原因及解決

    springboot 設(shè)置server.port不生效的原因及解決

    這篇文章主要介紹了springboot 設(shè)置server.port不生效的原因及解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-08-08
  • 剖析Java中線程編程的概念

    剖析Java中線程編程的概念

    這篇文章主要介紹了Java中線程編程的概念,是Java入門學(xué)習(xí)中的基礎(chǔ)知識(shí),需要的朋友可以參考下
    2015-09-09
  • 基于@AllArgsConstructor與@Value共用的問題解決

    基于@AllArgsConstructor與@Value共用的問題解決

    這篇文章主要介紹了基于@AllArgsConstructor與@Value共用的問題解決,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-09-09
  • Java導(dǎo)出Word文檔的四種方法

    Java導(dǎo)出Word文檔的四種方法

    在日常的開發(fā)工作中,我們時(shí)常會(huì)遇到導(dǎo)出Word文檔報(bào)表的需求,比如公司的財(cái)務(wù)報(bào)表、醫(yī)院的患者統(tǒng)計(jì)報(bào)表、電商平臺(tái)的銷售報(bào)表等等,所以本文給大家介紹了Java導(dǎo)出Word文檔的四種方法,并通過代碼示例講解的非常詳細(xì),需要的朋友可以參考下
    2025-03-03
  • Java實(shí)現(xiàn)順序表的增刪查改功能

    Java實(shí)現(xiàn)順序表的增刪查改功能

    這篇文章主要介紹了Java實(shí)現(xiàn)順序表的增刪查改功能,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2021-04-04
  • SpringBoot項(xiàng)目中feignClient使用方式

    SpringBoot項(xiàng)目中feignClient使用方式

    文章介紹了在Spring Boot項(xiàng)目中配置Feign客戶端攔截器的具體步驟,包括在application.yml中添加配置、在主類上啟用組件掃描、將攔截器加入到攔截器列表中以及在接口調(diào)用時(shí)的說明,總結(jié)指出這是個(gè)人經(jīng)驗(yàn)分享,希望對(duì)大家有所幫助
    2024-11-11
  • Spring Boot中使用RabbitMQ 生產(chǎn)消息和消費(fèi)消息的實(shí)例代碼

    Spring 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-10
  • QR 二維碼中插入圖片實(shí)現(xiàn)方法

    QR 二維碼中插入圖片實(shí)現(xiàn)方法

    這篇文章主要介紹了QR 二維碼中插入圖片實(shí)現(xiàn)方法的相關(guān)資料,需要的朋友可以參考下
    2016-11-11
  • Java多線程中的wait、notify和park、unpark的使用詳解

    Java多線程中的wait、notify和park、unpark的使用詳解

    這篇文章主要介紹了Java多線程中的wait、notify和park、unpark的使用詳解,它們都是線程之間進(jìn)行協(xié)作的手段,都屬于 Object 對(duì)象的方法,必須獲得此對(duì)象的鎖,才能調(diào)用這幾個(gè)方法,需要的朋友可以參考下
    2023-12-12

最新評(píng)論