@Conditional注解 -【Spring底层原理

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

// Linux系统判断条件

public class LinuxCondition implements Condition {

/**

  • @description 判断操作系统是否是Linux系统

  • @author ONESTAR

  • @date 2021/2/10 10:56

  • @param conditionContext

  • @param annotatedTypeMetadata

  • @throws

  • @return boolean

*/

public boolean matches(ConditionContext conditionContext, AnnotatedTypeMetadata annotatedTypeMetadata) {

// 获取当前环境

Environment environment = conditionContext.getEnvironment();

// 判断是否是Linux系统

String property = environment.getProperty(“os.name”);

if (property.contains(“linux”)){

return true;

}

return false;

}

}

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

@Configuration

public class AppConfig {

// 如果WindowsCondition的实现方法返回true,则注入这个bean

@Conditional({WindowsCondition.class})

@Bean

public User user1(){

return new User();

}

// 如果LinuxCondition的实现方法返回true,则注入这个bean

@Conditional({LinuxCondition.class})

@Bean

public User user2(){

return new User();

}

}

这时我们再来运行启动类,默认情况下是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

public boolean shouldSkip(@Nullable AnnotatedTypeMetadata metadata, @Nullable ConfigurationPhase phase) {

// 检查注解中是否包含@Conditional类型的注解

if (metadata != null && metadata.isAnnotated(Conditional.class.getName())) {

// 判断当前bean是解析还是注册

if (phase == null) {

// bean的注解信息封装对象是AnnotationMetadata类型并且,类上有@Component,@ComponentScan,@Import,@ImportResource,则表示为解析类型

return metadata instanceof AnnotationMetadata && ConfigurationClassUtils.isConfigurationCandidate((AnnotationMetadata)metadata) ? this.shouldSkip(metadata, ConfigurationPhase.PARSE_CONFIGURATION) : this.shouldSkip(metadata, ConfigurationPhase.REGISTER_BEAN);

} else {

List conditions = new ArrayList();

Iterator var4 = this.getConditionClasses(metadata).iterator();

// 从bean的注解信息封装对象中获取所有的Conditional类型或者Conditional的派生注解

while(var4.hasNext()) {

String[] conditionClasses = (String[])var4.next();

String[] var6 = conditionClasses;

int var7 = conditionClasses.length;

for(int var8 = 0; var8 < var7; ++var8) {

String conditionClass = var6[var8];

// 实例化Conditional中的条件判断类(Condition的子类)

Condition condition = this.getCondition(conditionClass, this.context.getClassLoader());

// 添加到条件集合中

conditions.add(condition);

}

}

// 根据Condition的优先级进行排序

AnnotationAwareOrderComparator.sort(conditions);

var4 = conditions.iterator();

Condition condition;

ConfigurationPhase requiredPhase;

do {

do {

if (!var4.hasNext()) {

return false;

}

condition = (Condition)var4.next();

requiredPhase = null;

// 如果是ConfigurationCondition类型的Condition

if (condition instanceof ConfigurationCondition) {

// 获取需要对bean进行的操作,是解析还是注册

requiredPhase = ((ConfigurationCondition)condition).getConfigurationPhase();

}

//(如果requiredPhase==null或者指定的操作类型是目前阶段的操作类型)并且不符合设置的条件则跳过

} while(requiredPhase != null && requiredPhase != phase);

} while(condition.matches(this.context, metadata));

return true;

}

} else {

return false;

}

}

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

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

【2】ConfigurationClassPostProcessorprocessConfigBeanDefinitions

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

public void processConfigBeanDefinitions(BeanDefinitionRegistry registry) {

List configCandidates = new ArrayList();

// 获取registry中定义的所有的bean的name

String[] candidateNames = registry.getBeanDefinitionNames();

do {

// 第一个会调用shouldSkip的位置,这里是解析能够直接获取的候选配置bean。可能是Component,ComponentScan,Import,ImportResource或者有Bean注解的bean

StartupStep processConfig = this.applicationStartup.start(“spring.context.config-classes.parse”);

parser.parse(candidates);

parser.validate();

// 获取上面封装已经解析过的配置bean的ConfigurationClass集合

Set configClasses = new LinkedHashSet(parser.getConfigurationClasses());

// 移除前面已经处理过的

configClasses.removeAll(alreadyParsed);

if (this.reader == null) {

this.reader = new ConfigurationClassBeanDefinitionReader(registry, this.sourceExtractor, this.resourceLoader, this.environment, this.importBeanNameGenerator, parser.getImportRegistry());

}

//第二个会调用shouldSkip的位置,这里是加载configurationClasse中内部可能存在配置bean,比如方法上加了@Bean或者@Configuration标签的bean

this.reader.loadBeanDefinitions(configClasses);

alreadyParsed.addAll(configClasses);

}

}

  • 通过parse方法解析BeanDefinitionRegistry中能直接获取到的候选bean,并解析保存到ConfigurationClassParser类的保存解析过的配置类的集合configurationClasses
  • loadBeanDefinitions则是对上面解析的集合configurationClasses中的bean内部的进一步的处理,处理类内部定义的bean

【3】ConfigurationClassParserparse方法

public void parse(Set configCandidates) {

Iterator var2 = configCandidates.iterator();

while(var2.hasNext()) {

先自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Java工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《Java开发全套学习资料》送给大家,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

img

img

img

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频

如果你觉得这些内容对你有帮助,可以扫码领取!

img

最后

分享一套我整理的面试干货,这份文档结合了我多年的面试官经验,站在面试官的角度来告诉你,面试官提的那些问题他最想听到你给他的回答是什么,分享出来帮助那些对前途感到迷茫的朋友。

面试经验技巧篇
  • 经验技巧1 如何巧妙地回答面试官的问题
  • 经验技巧2 如何回答技术性的问题
  • 经验技巧3 如何回答非技术性问题
  • 经验技巧4 如何回答快速估算类问题
  • 经验技巧5 如何回答算法设计问题
  • 经验技巧6 如何回答系统设计题
  • 经验技巧7 如何解决求职中的时间冲突问题
  • 经验技巧8 如果面试问题曾经遇见过,是否要告知面试官
  • 经验技巧9 在被企业拒绝后是否可以再申请
  • 经验技巧10 如何应对自己不会回答的问题
  • 经验技巧11 如何应对面试官的“激将法”语言
  • 经验技巧12 如何处理与面试官持不同观点这个问题
  • 经验技巧13 什么是职场暗语

面试真题篇
  • 真题详解1 某知名互联网下载服务提供商软件工程师笔试题
  • 真题详解2 某知名社交平台软件工程师笔试题
  • 真题详解3 某知名安全软件服务提供商软件工程师笔试题
  • 真题详解4 某知名互联网金融企业软件工程师笔试题
  • 真题详解5 某知名搜索引擎提供商软件工程师笔试题
  • 真题详解6 某初创公司软件工程师笔试题
  • 真题详解7 某知名游戏软件开发公司软件工程师笔试题
  • 真题详解8 某知名电子商务公司软件工程师笔试题
  • 真题详解9 某顶级生活消费类网站软件工程师笔试题
  • 真题详解10 某知名门户网站软件工程师笔试题
  • 真题详解11 某知名互联网金融企业软件工程师笔试题
  • 真题详解12 国内某知名网络设备提供商软件工程师笔试题
  • 真题详解13 国内某顶级手机制造商软件工程师笔试题
  • 真题详解14 某顶级大数据综合服务提供商软件工程师笔试题
  • 真题详解15 某著名社交类上市公司软件工程师笔试题
  • 真题详解16 某知名互联网公司软件工程师笔试题
  • 真题详解17 某知名网络安全公司校园招聘技术类笔试题
  • 真题详解18 某知名互联网游戏公司校园招聘运维开发岗笔试题

资料整理不易,点个关注再走吧
某初创公司软件工程师笔试题

  • 真题详解7 某知名游戏软件开发公司软件工程师笔试题
  • 真题详解8 某知名电子商务公司软件工程师笔试题
  • 真题详解9 某顶级生活消费类网站软件工程师笔试题
  • 真题详解10 某知名门户网站软件工程师笔试题
  • 真题详解11 某知名互联网金融企业软件工程师笔试题
  • 真题详解12 国内某知名网络设备提供商软件工程师笔试题
  • 真题详解13 国内某顶级手机制造商软件工程师笔试题
  • 真题详解14 某顶级大数据综合服务提供商软件工程师笔试题
  • 真题详解15 某著名社交类上市公司软件工程师笔试题
  • 真题详解16 某知名互联网公司软件工程师笔试题
  • 真题详解17 某知名网络安全公司校园招聘技术类笔试题
  • 真题详解18 某知名互联网游戏公司校园招聘运维开发岗笔试题

[外链图片转存中…(img-hUyF5YYS-1711194455987)]

资料整理不易,点个关注再走吧
需要更多Java资料的小伙伴可以帮忙点赞+关注,点击传送门,即可免费领取!

  • 28
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值