当前位置: 首页 > news >正文

【Spring Boot 源码学习】@Conditional 条件注解

Spring Boot 源码学习系列

在这里插入图片描述

@Conditional 条件注解

  • 引言
  • 往期内容
  • 主要内容
    • 1. 初识 @Conditional
    • 2. @Conditional 的衍生注解
  • 总结

引言

前面的博文,Huazie 带大家从 Spring Boot 源码深入了解了自动配置类的读取和筛选的过程,然后又详解了OnClassConditionOnBeanConditionOnWebApplicationCondition 这三个自动配置过滤匹配子类实现。

在上述的博文中,我们其实已经初步涉及到了像 @ConditionalOnClass@ConditionalOnBean@ConditionalOnWebApplication 这样的条件注解,并且这些条件注解里面,我们都能看到 @Conditional 注解。

在这里插入图片描述

往期内容

在开始本篇的内容介绍之前,我们先来看看往期的系列文章【有需要的朋友,欢迎关注系列专栏】:

Spring Boot 源码学习
Spring Boot 项目介绍
Spring Boot 核心运行原理介绍
【Spring Boot 源码学习】@EnableAutoConfiguration 注解
【Spring Boot 源码学习】@SpringBootApplication 注解
【Spring Boot 源码学习】走近 AutoConfigurationImportSelector
【Spring Boot 源码学习】自动装配流程源码解析(上)
【Spring Boot 源码学习】自动装配流程源码解析(下)
【Spring Boot 源码学习】深入 FilteringSpringBootCondition
【Spring Boot 源码学习】OnClassCondition 详解
【Spring Boot 源码学习】OnBeanCondition 详解
【Spring Boot 源码学习】OnWebApplicationCondition 详解

主要内容

本篇我们重点介绍 @Conditional 条件注解,参见如下:

1. 初识 @Conditional

我们先来看看 @Conditional 注解的源码【Spring Context 5.3.25】:

/*** 表示组件仅在所有指定条件匹配时才有资格注册。* * 条件是在bean定义即将注册之前可以通过编程确定的任何状态(有关详细信息,请参阅Condition)。* * @Conditional注解可以以以下任意方式使用:* 	作为类型级别的注释直接或间接地应用于带有@Component的任何类,包括@Configuration类* 	作为元注释,用于组合自定义注释标签* 	作为@Bean方法上的注释级别注解* * 如果一个@Configuration类被标记为@Conditional,则该类的所有@Bean方法、@Import注解和@ComponentScan注解都将受到条件约束。** @author Phillip Webb* @author Sam Brannen* @since 4.0* @see Condition*/
@Target({ElementType.TYPE, ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface Conditional {/*** 必须匹配才能注册组件的所有条件类*/Class<? extends Condition>[] value();}

翻看上述源码,可以看到 @Conditional 条件注解是从 Spring 4.0 开始引入的,它表示组件仅在所有指定条件匹配时才有资格注册。比如,当类加载器下存在某个指定的类的时候才会对注解的类进行实例化操作。

它唯一的元素属性是接口 Condition 的数组,只有数组中指定的所有 Conditionmatches 方法都返回 true 的情况下,被注解的类才会被加载。我们前面讲到的 OnClassCondition 等类就是 Condition 的子类之一。

/*** 一个必须匹配才能注册的单个 Condition。** <p> 在 bean 定义即将被注册之前立即进行检查,并可以根据在该点可以确定的任何标准自由否决注册。** <p> 条件必须遵循与 BeanFactoryPostProcessor 相同的限制,并确保不要与 bean 实例进行交互。* 对于与 @Configuration beans交互的更细粒度的控制,请考虑实现 ConfigurationCondition 接口。** @author Phillip Webb* @since 4.0* @see ConfigurationCondition* @see Conditional* @see ConditionContext*/
@FunctionalInterface
public interface Condition {/*** 确定条件是否匹配。* @param context 条件上下文* @param metadata 正在检查的 AnnotationMetadata 或 MethodMetadata 的元数据* @return 如果条件匹配并且可以注册组件,则返回 true;否则返回 false,否决带有注解的组件的注册。*/boolean matches(ConditionContext context, AnnotatedTypeMetadata metadata);}

上述就是 Condition 接口的源码,它的 matches 方法用来确定条件是否匹配,其中两个参数分别如下:

  • ConditionContext :条件上下文,可通过该接口提供的方法来获得 Spring 应用的上下文信息,接口定义如下:

    public interface ConditionContext {/*** 返回一个 BeanDefinitionRegistry 对象,该对象将包含如果条件匹配时应该持有的bean定义。* 如果没有可用的注册表(这种情况很少见:只有当使用 ClassPathScanningCandidateComponentProvider 时才会出现),* 则会抛出IllegalStateException异常。*/BeanDefinitionRegistry getRegistry();/*** 返回一个 ConfigurableListableBeanFactory 对象,该对象将包含如果条件匹配时应该持有的bean定义,* 或者 如果bean工厂不可用(或者无法向下转型为 ConfigurableListableBeanFactory),则返回null。*/@NullableConfigurableListableBeanFactory getBeanFactory();/*** 返回当前应用程序正在运行的环境。*/Environment getEnvironment();/*** 返回当前正在使用的资源加载器。*/ResourceLoader getResourceLoader();/*** 返回应该用来加载额外类的 ClassLoader。如果系统类加载器不可访问,则返回null。*/@NullableClassLoader getClassLoader();}
    
  • AnnotatedTypeMetadata :该接口提供了访问特定类或方法的注解功能,并且不需要加载类,可以用来检查带有 @Bean 注解的方法上是否还有其他注解。

    下面我们来查看下它的源码【spring-core 5.3.25】:

    public interface AnnotatedTypeMetadata {// 返回一个MergedAnnotations对象,表示该类型的注解集合。MergedAnnotations getAnnotations();// 检查是否存在指定名称的注解,如果存在则返回true,否则返回false。default boolean isAnnotated(String annotationName) {return this.getAnnotations().isPresent(annotationName);}// 下面的方法,都是用来获取指定名称注解的属性值@Nullabledefault Map<String, Object> getAnnotationAttributes(String annotationName) {return this.getAnnotationAttributes(annotationName, false);}@Nullabledefault Map<String, Object> getAnnotationAttributes(String annotationName, boolean classValuesAsString) {MergedAnnotation<Annotation> annotation = this.getAnnotations().get(annotationName, (Predicate)null, MergedAnnotationSelectors.firstDirectlyDeclared());return !annotation.isPresent() ? null : annotation.asAnnotationAttributes(Adapt.values(classValuesAsString, true));}@Nullabledefault MultiValueMap<String, Object> getAllAnnotationAttributes(String annotationName) {return this.getAllAnnotationAttributes(annotationName, false);}@Nullabledefault MultiValueMap<String, Object> getAllAnnotationAttributes(String annotationName, boolean classValuesAsString) {Adapt[] adaptations = Adapt.values(classValuesAsString, true);return (MultiValueMap)this.getAnnotations().stream(annotationName).filter(MergedAnnotationPredicates.unique(MergedAnnotation::getMetaTypes)).map(MergedAnnotation::withNonMergedAttributes).collect(MergedAnnotationCollectors.toMultiValueMap((map) -> {return map.isEmpty() ? null : map;}, adaptations));}
    }
    

2. @Conditional 的衍生注解

Spring Bootautoconfigure 项目中提供了各类基于@Conditional 注解的衍生注解,它们均位于 spring-boot-autoconfigure 项目的 org.springframework.boot.autoconfigure.condition 包下,如下图所示:。

在这里插入图片描述

上述有好几个条件注解,我们已经接触过了,下面我们再仔细介绍一下:

  • @ConditionalOnBean:当容器中有指定 Bean 的条件下。

  • @ConditionalOnClass:当 classpath 类路径下有指定类的条件下。

  • @ConditionalOnCloudPlatform:当指定的云平台处于 active 状态时。

  • @ConditionalOnExpression:基于 SpEL 表达式的条件判断。

  • @ConditionalOnJava:基于 JVM 版本作为判断条件。

  • @ConditionalOnJndi:在 JNDI 存在的条件下查找指定的位置。

  • @ConditionalOnMissingBean:当容器里没有指定 Bean 的条件。

  • @ConditionalOnMissingClass:当类路径下没有指定类的条件下。

  • @ConditionalOnNotWebApplication:当项目不是一个 Web 项目的条件下。

  • @ConditionalOnProperty:当指定的属性有指定的值的条件下。

  • @ConditionalOnResource:类路径是否有指定的值。

  • @ConditionalOnSingleCandidate:当指定的 Bean 在容器中只有一个,或者有多个但是指定了首选的 Bean

  • @ConditionalOnWarDeployment :当应用以 War 包形式部署时(例如在 Tomcat、Jetty 等 Web 服务器中)

  • @ConditionalOnWebApplication:当项目是一个 Web 项目的条件下

如果我们仔细观察这些注解的源码,很快会发现它们其实都组合了@Conditional 注解,不同的是它们在注解中指定的条件(Condition)不同。

下面我们以前面博文中了解过的 @ConditionalOnWebApplication 为例,来对衍生条件注解进行一个简单的分析:

/*** 用于条件性地匹配应用程序是否为Web应用程序。默认情况下,任何Web应用程序都会匹配,但可以通过type()属性进行缩小范围。** @author Dave Syer* @author Stephane Nicoll* @since 1.0.0*/
@Target({ ElementType.TYPE, ElementType.METHOD })
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Conditional(OnWebApplicationCondition.class)
public @interface ConditionalOnWebApplication {// 所需的web应用类型Type type() default Type.ANY;// 可选应用类型枚举enum Type {	// 任何类型ANY,// 基于servlet的web应用SERVLET,// 基于reactive的web应用REACTIVE}
}

通过查看 @ConditionalOnWebApplication 注解的源码,我们发现它的确组合了 @Conditional 注解,并且指定了对应的 ConditionOnWebApplicationCondition。该类继承自 SpringBootCondition 并实现 AutoConfigurationImportFilter 接口。

有关 OnWebApplicationCondition 类的详细介绍,请查看笔者的《【Spring Boot 源码学习】OnWebApplicationCondition 详解》,

了解了条件类的相关内容后,我们可以用如下图来表示 Condition 接口相关功能及实现类:

在这里插入图片描述

总结

本篇我们介绍 @Conditional 条件注解及其衍生注解,至此有关自动配置装配的流程已经基本介绍完毕。

虽然我们从源码角度对自动装配流程有了清晰的认识,但还是不能熟练地运用。那么下篇博文,我们将以 Spring Boot 内置的 http 编码功能为例来分析一下整个自动配置的过程。

相关文章:

【Spring Boot 源码学习】@Conditional 条件注解

Spring Boot 源码学习系列 Conditional 条件注解 引言往期内容主要内容1. 初识 Conditional2. Conditional 的衍生注解 总结 引言 前面的博文&#xff0c;Huazie 带大家从 Spring Boot 源码深入了解了自动配置类的读取和筛选的过程&#xff0c;然后又详解了OnClassCondition、…...

jupyter_快速开始

文章目录 使用 Anaconda 启动 jupyter-lab纯 python 环境使用 jupyter-notebook纯 python 环境使用 jupyter-labjupyter-lab 配置文件相关jupyter-notebook 配置文件相关jupyter-lab 与 jupyter-notebook 的关系与区别 使用 Anaconda 启动 jupyter-lab 启动一个cmd 命令行&…...

英特尔 SGX 技术概述

目录 介绍概述指示结构Memory安全区页面缓存Enclave Page Cache &#xff08;EPC&#xff09;安全区页面缓存映射Enclave Page Cache Map (EPCM) Memory ManagementStructures页面信息Page Information (PAGEINFO)安全信息Security Information (SECINFO)分页加密元数据Paging …...

SpringBoot核心功能与基础配置

SpringBoot简介 原先的Spring程序缺点&#xff0c;包括依赖设置繁琐&#xff0c;每项jar的引用都需要自己撰写。并且配置繁琐&#xff0c;配置文件中也需要自己写加载bean等。由此针对原始的Spring程序&#xff0c;Pivotal团队提供的全新框架——SpringBoot&#xff0c;其设计…...

vue3后台管理框架之Mock开发

前言 在前后端对接中&#xff0c;有时后端的接口数据没有 那么快能给出&#xff0c;因此我们可以通过mock模拟自己的请求数据&#xff0c;在后端接口没有给出的同时&#xff0c;先使用mock请求的数据完成前端相关的逻辑 官方文档&#xff1a;vite-plugin-mock vite 的数据模…...

03_51单片机点亮LED灯

51单片机是一种非常常见的单片机型号&#xff0c;广泛应用于各种嵌入式系统和电子设备中。LED灯是一种常见的输出设备&#xff0c;用于显示信息或指示状态。下面是关于51单片机控制LED灯的介绍&#xff1a; 1. 连接LED灯&#xff1a;将LED的正极连接到51单片机的一个I/O引脚&a…...

【前端设计模式】之备忘录模式

备忘录模式是一种行为设计模式&#xff0c;它允许在不破坏封装性的前提下捕获和恢复对象的内部状态。在前端开发中&#xff0c;备忘录模式可以用于保存和恢复用户界面的状态&#xff0c;以及实现撤销和重做功能。 备忘录模式特性&#xff1a; 封装了对象的状态&#xff1a;备…...

复习Day15:栈与队列part02:20. 有效的括号、1047.删除字符串中所有相邻重复项

我用的方法是在leetcode再过一遍例题&#xff0c;明显会的就复制粘贴&#xff0c;之前没写出来就重写&#xff0c;然后从拓展题目中找题目来写。辅以Labuladong的文章看。然后刷题不用CLion了&#xff0c;使用leetcode自带的IDE模拟面试环境。 历史博客链接&#xff1a; http…...

基于Java的宠物商城管理系统设计与实现(源码+lw+部署文档+讲解等)

文章目录 前言具体实现截图论文参考详细视频演示为什么选择我自己的网站自己的小程序&#xff08;小蔡coding&#xff09;有保障的售后福利 代码参考源码获取 前言 &#x1f497;博主介绍&#xff1a;✌全网粉丝10W,CSDN特邀作者、博客专家、CSDN新星计划导师、全栈领域优质创作…...

Python的GIL存在的情况下,是否还有必要添加线程锁。

GIL锁的产生&#xff1a; 为了保证在单线程情况下&#xff0c;Python的正常执行和效率&#xff0c;GIL锁产生了&#xff0c;由于只有一把锁就不会产生死锁也不用切换。 对于Python语言而言&#xff0c;只有CPython解释器&#xff08;用C语言编写的Python解释库&#xff09;存在…...

基于下垂控制的孤岛双机并联逆变器环流抑制MATLAB仿真模型

微❤关注“电气仔推送”获得资料&#xff08;专享优惠&#xff09; 在实际应用中逆变器都是并联运行的,但是逆变器的并联运行也存在不少问题,由于线路阻抗差异、各个逆变器输出端瞬时电压幅值不同等,都容易导致环流的出现。环流会导致逆变器损耗增加,从而影响微电网的输出效率…...

spring事务面试题

1.Spring 事务实现方式有哪些&#xff1f; 事务就是一系列的操作原子操作,Spring事务机制主要 包括声明式事务和编程式事务。 编程式事务&#xff1a;通过编程的方式管理事务&#xff0c;自己设置未提交模式&#xff0c;自己获取连接&#xff0c;自己预编译&#xff0c;自己回…...

C++标准库算法整理

目录 1、数值操作 1.1、std::accumulate 1.2、std::inner_product 1.3、std::partial_sum 1.4、std::exclusive_scan 1.5、std::inclusive_scan 1.6、std::reduce 2、相邻元素 2.1、std::adjacent_difference 2.2、std::adjacent_find 2.3、std::unique 2.4、std::u…...

【Codeforces】Codeforces Round 903 (Div. 3)【待补】

Dashboard - Codeforces Round 903 (Div. 3) - Codeforces Problem - C - Codeforces Problem - D - Codeforces...

workerman 运行时报错 Call to undefined function posix_getpid()

使用 验证php扩展是否齐全 curl -Ss https://www.workerman.net/check | php缺少posix 下载 在 Linux 系统上&#xff0c;可以使用包管理器来安装 php-posix 扩展&#xff0c;例如 Ubuntu 系统可以通过以下命令进行安装&#xff1a; sudo apt-get install php-posix如果你使用…...

【探讨C++中的临时对象:一时之物还是永恒之道?】

在C编程中&#xff0c;临时对象是一个经常引起讨论的话题。它们是什么&#xff0c;为什么它们存在&#xff0c;以及如何正确使用它们&#xff1f;本文将深入探讨C中的临时对象&#xff0c;帮助您理解它们的含义和用途。 什么是临时对象&#xff1f; 临时对象&#xff08;Temp…...

二叉树相关算法

1、二叉树基本操作 二叉树的定义就不在这里多说了&#xff0c;下面这个图就是一个简单的二叉树&#xff1a; 二叉树的三种遍历方式&#xff1a; 前序遍历&#xff1a;头左右&#xff0c;也就是先头后左再右&#xff1a;1245367 public static void prePrint(BinaryTreeNode …...

Vue_Bug npm install报错 code:128

Bug描述&#xff1a; npm install报错 code&#xff1a;128 npm ERR! Warning: Permanently added ‘github.com’ (ED25519) to the list of known hosts. npm ERR! gitgithub.com: Permission denied (publickey). npm ERR! fatal: Could not read from remote repository. n…...

【Unity ShaderGraph】| 如何快速制作一个 马赛克效果 实战

前言 【Unity ShaderGraph】| 如何快速制作一个 马赛克效果 实战一、效果展示二、马赛克效果四、应用实例 前言 本文将使用Unity 的ShaderGraph制作一个马赛克的效果&#xff0c;可以直接拿到项目中使用。对ShaderGraph还不了解的小伙伴可以参考这篇文章&#xff1a;【Unity S…...

【Java 进阶篇】JavaScript DOM Document对象详解

在前端开发中&#xff0c;DOM&#xff08;文档对象模型&#xff09;扮演着重要的角色。它允许我们使用JavaScript来与网页文档进行交互&#xff0c;实现动态的网页效果。DOM的核心部分之一就是Document对象&#xff0c;它代表了整个HTML文档。在本篇博客中&#xff0c;我们将深…...

使用taotoken聚合api后c语言项目调用大模型的延迟与稳定性体验

&#x1f680; 告别海外账号与网络限制&#xff01;稳定直连全球优质大模型&#xff0c;限时半价接入中。 &#x1f449; 点击领取海量免费额度 使用taotoken聚合api后c语言项目调用大模型的延迟与稳定性体验 在开发基于C语言的服务器后台时&#xff0c;集成大模型能力常面临一…...

30 岁硕士 Linux C 开发背景,未来想去澳洲就业,研究方向该选 AI、SDN 漏洞还是 Linux 内核?

结合截至 2026 年 5 月 19 日的澳洲就业市场公开信息与岗位语境整理 这类题最容易把人带偏的一步&#xff0c;是一上来就把问题理解成&#xff1a; 哪条研究方向听起来最前沿。 但如果你的目标不是单纯做学术&#xff0c;而是&#xff1a; 未来去澳洲更容易就业。 那问题其…...

别再只会Hello World了!用Hadoop 3.x + Eclipse手把手搞定你的第一个MapReduce词频统计

从Hello World到实战&#xff1a;用Hadoop 3.x实现你的第一个词频统计项目 当你第一次接触编程时&#xff0c;"Hello World"可能是你学会的第一个程序。这个简单的程序让你理解了如何让计算机输出一段文字。但编程的世界远不止于此&#xff0c;特别是当你开始探索大数…...

2026,AI Agent 真的开始上班了——从 MCP 协议到生产部署,一份踩坑实录

爆款标题备选2026 年&#xff0c;我司来了一个 AI 同事——Agent 落地实录MCP 协议 LangChain Dify&#xff1a;把 AI Agent 塞进生产环境的正确姿势BBC 报道了三个中国人的 AI 恐惧&#xff0c;但我想说点不一样的AI Agent 从 Demo 到生产&#xff0c;中间隔着一个 MCP 协议…...

CARTGen-IR: Synthetic Tabular Data Generation for Imbalanced Regression——基于CART的表格数据不平衡回归合成采样方法

一、研究问题与背景 1.1 问题定义 不平衡回归&#xff1a;在连续目标变量中&#xff0c;极端值&#xff08;高值或低值&#xff09;样本稀少&#xff0c;导致模型偏向预测平均值&#xff0c;忽略重要极端情况。 应用场景&#xff1a;极端天气预测、海面温度异常、药物敏感性检…...

用Python和Matplotlib搞定高光谱图像可视化:从.mat文件到伪彩色图(附完整代码)

PythonMatplotlib高光谱图像可视化实战&#xff1a;从.mat文件到伪彩色合成 高光谱图像处理正逐渐从专业遥感领域走向更广泛的工业应用场景。当一位农业科技公司的算法工程师第一次拿到作物生长监测的高光谱数据时&#xff0c;面对.mat格式文件中那个神秘的三维矩阵&#xff0c…...

告别复杂推流!ESP32-CAM直连点灯APP实现视频监控,完整配置流程与源码解析

ESP32-CAM直连点灯APP&#xff1a;零服务器视频监控方案全解析 在物联网设备开发中&#xff0c;视频监控一直是个既诱人又令人头疼的功能。传统方案需要搭建RTSP/RTMP服务器&#xff0c;配置复杂的网络转发规则&#xff0c;这让很多开发者望而却步。而今天我们要介绍的ESP32-CA…...

毕业设计 基于python的答题卡识别评分系统

文章目录 0 简介课题简介什么是机器视觉实现步骤详细设计图片读取canny边缘检测四点变换 划出区域处理选择题区域提取选项轮廓判断选项 读取正确结果 最后 0 简介 今天学长向大家分享一个毕业设计项目 毕业设计 基于python的答题卡识别评分系统 项目运行效果&#xff1a; 毕…...

毕业设计作品精选【芳心科技】基于STM32的智能家庭快递柜

实物效果图&#xff1a;实现功能&#xff1a;本设计的基于STM32单片机的智能家庭快递柜&#xff0c;需要及进行硬件没计和软件开发。硬件方面&#xff0c;需要选择合适的矩阵键盘、显示器、LED灯、电动机等硬件没备&#xff0c;并设计相应的电路来连接各个模块。软件方面&#…...

NY378固态MT29F32T08GSLBHL8-24QA:B

NY378固态MT29F32T08GSLBHL8-24QA:B你是否曾好奇&#xff0c;那些默默支撑着工业设备稳定运行、保障数据高速流转的存储核心&#xff0c;究竟蕴藏着怎样的技术密码&#xff1f;今天&#xff0c;我们将聚焦一颗在特定领域中扮演关键角色的芯片——来自美光&#xff08;Micron&am…...