SpringBoot注解@ConditionalOnClass底層源碼實(shí)現(xiàn)
@ConditionalOnClass的底層源碼實(shí)現(xiàn)
在SpringBoot中,支持了很多種條件注解,@ConditionalOnClass注解就是其中之一,而且及其重要,它主要是用來判斷該注解所指定的某個(gè)類或某些類,是否在ClassPath中存在,如果存在則符合條件,如果不存在則不符合。
@Target({ ElementType.TYPE, ElementType.METHOD })
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Conditional(OnClassCondition.class)
public @interface ConditionalOnClass {
Class<?>[] value() default {};
String[] name() default {};
}
這是該注解的源碼,可以通過value和name來指定要判斷的類,而真正執(zhí)行判斷的邏輯在OnClassCondition類中。
- OnClassCondition類繼承了FilteringSpringBootCondition類
- FilteringSpringBootCondition類又繼承了SpringBootCondition類
- SpringBootCondition類實(shí)現(xiàn)了Condition接口

Spring在解析條件注解時(shí),就會(huì)調(diào)用Condition接口的matches()方法,在上面的類繼承關(guān)系中,SpringBootCondition類實(shí)現(xiàn)了matches()方法,所以會(huì)先被調(diào)用。
ConditionOutcome對(duì)象
在matches()方法中,會(huì)調(diào)用getMatchOutcome()方法,并得到ConditionOutcome對(duì)象,ConditionOutcome對(duì)象就表示條件判斷的結(jié)果。
public class ConditionOutcome {
// 表示條件是否匹配
private final boolean match;
// ...
}
getMatchOutcome()方法在SpringBootCondition類中是一個(gè)抽象方法,在子類OnClassCondition類中才真正實(shí)現(xiàn)了getMatchOutcome()方法,并真正會(huì)進(jìn)行條件判斷。
所以核心就是這個(gè)getMatchOutcome()方法,在這個(gè)方法中會(huì)先獲取@ConditionalOnClass注解的value和name屬性的值,這些值就是待判斷的類名集合。
// 調(diào)用getCandidates方法 List<String> onClasses = getCandidates(metadata, ConditionalOnClass.class);
private List<String> getCandidates(AnnotatedTypeMetadata metadata, Class<?> annotationType) {
MultiValueMap<String, Object> attributes =
metadata.getAllAnnotationAttributes(annotationType.getName(), true);
if (attributes == null) {
return null;
}
List<String> candidates = new ArrayList<>();
addAll(candidates, attributes.get("value"));
addAll(candidates, attributes.get("name"));
return candidates;
}
ClassNameFilter.MISSING判斷某類是否不存在
接下來就會(huì)逐個(gè)判斷類名集合中的每個(gè)類名,判斷邏輯為:利用ClassNameFilter.MISSING來判斷某類是否不存在?
List<String> missing = filter(onClasses, ClassNameFilter.MISSING, classLoader);
protected final List<String> filter(Collection<String> classNames, ClassNameFilter classNameFilter, ClassLoader classLoader) {
if (CollectionUtils.isEmpty(classNames)) {
return Collections.emptyList();
}
List<String> matches = new ArrayList<>(classNames.size());
for (String candidate : classNames) {
if (classNameFilter.matches(candidate, classLoader)) {
matches.add(candidate);
}
}
return matches;
}
ClassNameFilter.MISSING就是利用ClassLoader來加載類,如果加載到了表示類存在,沒加載到就表示不存在。
protected enum ClassNameFilter {
// ...
MISSING {
@Override
public boolean matches(String className, ClassLoader classLoader) {
// 是否不存在
return !isPresent(className, classLoader);
}
};
static boolean isPresent(String className, ClassLoader classLoader) {
if (classLoader == null) {
classLoader = ClassUtils.getDefaultClassLoader();
}
try {
resolve(className, classLoader);
return true;
}
catch (Throwable ex) {
return false;
}
}
}
protected static Class<?> resolve(String className, ClassLoader classLoader) throws ClassNotFoundException {
if (classLoader != null) {
return Class.forName(className, false, classLoader);
}
return Class.forName(className);
}
判斷完之后,只要missing集合不為空,那就表示待判斷的類中有類不存在,那就返回條件不匹配的ConditionOutcome對(duì)象,否則就返回條件匹配的ConditionOutcome對(duì)象。
這就是@ConditionalOnClass注解的核心源碼流程,期待你的點(diǎn)贊哦。
以上就是SpringBoot注解@ConditionalOnClass底層源碼實(shí)現(xiàn)的詳細(xì)內(nèi)容,更多關(guān)于SpringBoot ConditionalOnClass的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Intellij IDEA 最全超實(shí)用快捷鍵整理(長(zhǎng)期更新)
這篇文章主要介紹了Intellij IDEA 最全實(shí)用快捷鍵整理(長(zhǎng)期更新),本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-02-02
Java之while與do-while循環(huán)的用法詳解
在上一篇文章中,給大家講解了循環(huán)的概念,并重點(diǎn)給大家講解了for循環(huán)的使用。但在Java中,除了for循環(huán)之外,還有while、do-while、foreach等循環(huán)形式。這篇文章給大家講解while循環(huán)的使用2023-05-05
Java如何實(shí)現(xiàn)上傳文件到服務(wù)器指定目錄
這篇文章主要介紹了Java如何實(shí)現(xiàn)上傳文件到服務(wù)器指定目錄,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-04-04
java實(shí)現(xiàn)上傳圖片尺寸修改和質(zhì)量壓縮
這篇文章主要為大家詳細(xì)介紹了java實(shí)現(xiàn)上傳圖片尺寸修改和質(zhì)量壓縮,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-04-04
SpringBoot集成Validation參數(shù)校驗(yàn)
這篇文章主要為大家詳細(xì)介紹了SpringBoot集成Validation參數(shù)校驗(yàn),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-01-01
Java實(shí)現(xiàn)統(tǒng)計(jì)文件夾下所有文件的字?jǐn)?shù)
這篇文章主要為大家詳細(xì)介紹了如何使用Java實(shí)現(xiàn)統(tǒng)計(jì)文件夾下所有文件的字?jǐn)?shù),文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2024-03-03

