Spring Boot @Conditional注解
在Spring Boot中,@Conditional 注解用于条件性地注册bean。这意味着它可以根据某些条件来决定是否应该创建一个特定的bean。这个注解可以放在配置类或方法上,并且它会根据提供的一组条件来判断是否应该实例化对应的组件。
要使用 @Conditional注解时,需要实现 Condition 接口并重写 matches 方法。此方法将返回一个布尔值以指示条件是否匹配。如果条件为真,则创建bean;否则跳过该bean的创建。
以下是一个简单的例子,展示了如何使用自定义条件:
import org.springframework.context.annotation.Condition;
import org.springframework.context.annotation.ConditionContext;
import org.springframework.core.type.AnnotatedTypeMetadata;public class MyCustomCondition implements Condition {@Overridepublic boolean matches(ConditionContext context, AnnotatedTypeMetadata metadata) {// 在这里添加你的条件逻辑// 例如,检查系统属性、环境变量、已经存在的beans等return false; // 根据条件逻辑返回true或false}
}
- ConditionContext:提供了对当前解析上下文的访问,包括:
- Environment:可以用来获取环境变量、系统属性等。
- BeanFactory:如果可用的话,可以通过它访问已经注册的bean。
- ClassLoader:可以用来检查类路径上的类是否存在。
- EvaluationContext:可以用来评估SpEL表达式。
- AnnotatedTypeMetadata 提供了对带有注解的方法或类元数据的访问,例如注解属性值。
自定义条件类
假设我们有一个应用程序,它应该根据操作系统的不同来决定是否加载特定的bean。我们可以创建一个名为 OnWindowsCondition 的条件类:
import org.springframework.context.annotation.Condition;
import org.springframework.context.annotation.ConditionContext;
import org.springframework.core.type.AnnotatedTypeMetadata;public class OnWindowsCondition implements Condition {@Overridepublic boolean matches(ConditionContext context, AnnotatedTypeMetadata metadata) {return "win".equals(context.getEnvironment().getProperty("os.name").toLowerCase().substring(0, 3));}
}
然后在配置类中使用:
@Configuration
public class MyConfig {@Bean@Conditional(OnWindowsCondition.class)public WindowsSpecificService windowsSpecificService() {return new WindowsSpecificServiceImpl();}
}
Spring Boot提供内置条件注解
@ConditionalOnProperty
当你希望基于配置文件中的属性是否存在或者具有特定值来创建bean时,可以使用 @ConditionalOnProperty 注解。例如:
@Configuration
public class MyConfig {@Bean@ConditionalOnProperty(name = "my.feature.enabled", havingValue = "true")public MyFeature myFeature() {return new MyFeature();}
}
在这个例子中,只有当配置文件中存在名为 my.feature.enabled 的属性且其值为 true 时,才会创建 MyFeature bean。
更多用法
- prefix:指定属性名前缀。
- name:指定属性名(可以是数组,表示多个属性)。
- havingValue:指定属性必须具有的值,默认为空字符串。
- matchIfMissing:如果未找到属性,则默认匹配与否,默认为
false。
例如,你可以这样配置:
@Bean
@ConditionalOnProperty(prefix = "app", name = "feature.enabled", havingValue = "true", matchIfMissing = false)
public FeatureService featureService() {return new FeatureServiceImpl();
}
这将确保只有在 app.feature.enabled=true 时才会创建 FeatureService bean;如果没有设置该属性且 matchIfMissing=false,则不会创建。
@ConditionalOnClass 和 @ConditionalOnMissingClass
这两个注解用于检查类路径下是否存在或不存在某些类。这在集成第三方库时非常有用,因为你可以有条件地注册与这些库相关的bean。例如:
@Configuration
@ConditionalOnClass(name = "com.example.ExternalLibraryClass")
public class ExternalLibraryConfig {// ...
}
如果类路径中存在 ExternalLibraryClass 类,则会应用此配置。
@ConditionalOnBean 和 @ConditionalOnMissingBean
这些注解用于根据上下文中是否存在指定类型的bean来决定是否创建新的bean。这对于确保不会重复注册相同功能的bean非常有用。
@Bean
@ConditionalOnMissingBean(MyService.class)
public MyService myService() {return new MyServiceImpl();
}
这里的意思是:如果上下文中还没有类型为 MyService 的bean,则创建一个新的 MyServiceImpl 实例并注册为bean。
使用 SpEL 表达式的 @ConditionalOnExpression
可以通过 @ConditionalOnExpression 来编写复杂的条件表达式。例如,基于多个属性组合或者环境变量来决定是否创建bean。
@Bean
@ConditionalOnExpression("${spring.application.name:'default'} == 'myapp' && ${env:dev} == 'prod'")
public ProdSpecificBean prodSpecificBean() {return new ProdSpecificBean();
}
这段代码意味着只有当应用程序名称为 'myapp' 并且环境变量 env 设置为 'prod' 时,才会创建 ProdSpecificBean。
使用场景
动态条件评估
有时你可能需要在应用启动后根据某些变化(如用户输入或外部服务的状态)来动态调整bean的行为。虽然 @Conditional 主要用于启动时的静态条件判断,但你可以通过结合其他机制(如事件监听器、定时任务等)来实现类似的效果。
@Configuration
public class DynamicConditionConfig {private final AtomicBoolean shouldCreateBean = new AtomicBoolean(false);@Bean@ConditionalOnProperty(name = "dynamic.bean.enabled", havingValue = "true")public MyDynamicBean myDynamicBean() {return () -> shouldCreateBean.get();}// 模拟外部触发更新条件状态的方法public void updateCondition(boolean value) {shouldCreateBean.set(value);}
}
在这个例子中,MyDynamicBean 的行为依赖于一个原子布尔变量 shouldCreateBean,该变量可以在运行时被更改,从而影响bean的行为。
条件化的AOP切面
你还可以将条件应用于AOP切面,以实现更加灵活的横切关注点管理。例如:
@Aspect
@ConditionalOnProperty(name = "app.logging.enabled", havingValue = "true")
public class LoggingAspect {@Around("execution(* com.example.service.*.*(..))")public Object logExecutionTime(ProceedingJoinPoint joinPoint) throws Throwable {long start = System.currentTimeMillis();Object proceed = joinPoint.proceed();long executionTime = System.currentTimeMillis() - start;System.out.println(joinPoint.getTarget().getClass().getName() + "." + joinPoint.getSignature().getName() + " executed in " + executionTime + "ms");return proceed;}
}
这段代码表示只有当 app.logging.enabled=true 时才会激活日志记录切面。
使用 @Profile 和 @Conditional 结合
有时候,你可能想要结合 @Profile 和 @Conditional 来创建更精细的条件逻辑。例如:
@Configuration
@Profile("dev")
public class DevConfig {@Bean@ConditionalOnProperty(name = "feature.x.enabled", havingValue = "true")public FeatureX featureX() {return new FeatureXImpl();}
}
这里的意思是:仅在开发环境(dev profile)并且 feature.x.enabled=true 时才创建 FeatureX bean。
条件化代理
对于那些需要延迟初始化或者懒加载的bean,可以考虑使用 @Scope("proxy") 和 @Lazy 注解,结合 @Conditional 来实现条件化代理。
@Bean
@Scope(proxyMode = ScopedProxyMode.TARGET_CLASS)
@Lazy
@ConditionalOnProperty(name = "lazy.init.feature", havingValue = "true")
public LazyInitFeature lazyInitFeature() {return new LazyInitFeatureImpl();
}
这确保了只有在满足条件且首次访问 lazyInitFeature bean时才会实例化它。
调试技巧
使用 @PostConstruct 和 @PreDestroy 监控bean生命周期
为了更好地理解哪些bean被创建或销毁,可以在bean类中添加 @PostConstruct 和 @PreDestroy 方法,并输出日志信息。
@Component
@ConditionalOnProperty(name = "feature.y.enabled", havingValue = "true")
public class FeatureY {@PostConstructpublic void init() {System.out.println("FeatureY initialized.");}@PreDestroypublic void destroy() {System.out.println("FeatureY destroyed.");}
}
这种方法有助于跟踪bean的生命周期,并确认条件是否按预期工作。
使用 spring.main.banner-mode=off 减少干扰
当你专注于调试条件逻辑时,关闭Spring Boot启动横幅可以帮助减少不必要的输出,使日志更加清晰。
spring.main.banner-mode=off
常见问题
环境属性未正确加载
如果发现条件注解没有按照预期工作,请检查是否正确加载了环境属性文件(如 application.properties 或 application.yml)。确保这些文件位于正确的路径下,并且包含所需的属性定义。
类路径冲突
当遇到条件注解不起作用的问题时,类路径冲突是一个常见的原因。特别是当你使用 @ConditionalOnClass 或 @ConditionalOnMissingClass 时,确保项目中不存在重复的依赖项。你可以使用Maven或Gradle命令来分析依赖树:
- Maven:
mvn dependency:tree - Gradle:
gradle dependencies
条件逻辑错误
仔细审查你的条件逻辑,确保它们符合预期。可以通过单元测试验证每个条件的行为。例如:
@Test
void testFeatureYEnabled() {ApplicationContextRunner runner = new ApplicationContextRunner().withPropertyValues("feature.y.enabled=true");runner.run(context -> assertThat(context).hasSingleBean(FeatureY.class));
}@Test
void testFeatureYDisabled() {ApplicationContextRunner runner = new ApplicationContextRunner().withPropertyValues("feature.y.enabled=false");runner.run(context -> assertThat(context).doesNotHaveBean(FeatureY.class));
}
注意事项
- 条件注解只适用于Spring的配置阶段,因此它们不能用于运行时决策。
- 当使用
@Conditional或其他条件注解时,请确保你的条件逻辑不会导致循环依赖或意外的行为。 - 在编写条件逻辑时,考虑到性能影响,尽量使条件判断轻量级。
相关文章:
Spring Boot @Conditional注解
在Spring Boot中,Conditional 注解用于条件性地注册bean。这意味着它可以根据某些条件来决定是否应该创建一个特定的bean。这个注解可以放在配置类或方法上,并且它会根据提供的一组条件来判断是否应该实例化对应的组件。 要使用 Conditional注解时&#…...
jpeg文件学习
相关最全的一篇文章链接:https://www.cnblogs.com/wtysos11/p/14089482.html YUV基础知识 Y表示亮度分量:如果只显示Y的话,图像看起来会是一张黑白照。 U(Cb)表示色度分量:是照片蓝色部分去掉亮度&#x…...
c++基于过程
前言: 笔记基于C黑马程序员网课视频:黑马程序员匠心之作|C教程从0到1入门编程,学习编程不再难_哔哩哔哩_bilibili 在此发布笔记,只是为方便学习,不做其他用途,原作者为黑马程序员。 1. C基础 1.1 用Visual Studio写C程…...
FOC软件 STM32CubeMX 使用
1、安装-及相关软件版本 展示版本注意事项:keil MDK和STM32CubeMX版本至少要大于等于图中版本。 2、 Motor Profiler 5.2.0使用方法...
leetcode hot 100 全排列
46. 全排列 已解答 中等 相关标签 相关企业 给定一个不含重复数字的数组 nums ,返回其 所有可能的全排列 。你可以 按任意顺序 返回答案。 class Solution(object): def permute(self, nums): """ :type nums: List[int] :rtype: List[List[int…...
使用qrcode.vue生成当前网页的二维码(H5)
使用npm: npm install qrcode.vue 使用yarn: yarn add qrcode.vue package.json: 实现: <template><div class"code"><qrcode-vue :value"currentUrl" :size"size" render-as&…...
0055. shell命令--useradd
目录 55. shell命令--useradd 功能说明 语法格式 选项说明 选项 退出值 相关文件 /etc/passwd /etc/shadow /etc/group /etc/gshadow /etc/skel/ /etc/login.defs /etc/default/useradd 实践操作 注意事项 55. shell命令--useradd 功能说明 useradd 命令是 Lin…...
blender中合并的模型,在threejs中显示多个mesh;blender多材质烘培成一个材质
描述:在blender中合并的模型导出为glb,在threejs中导入仍显示多个mesh,并不是统一的整体,导致需要整体高亮或者使用DragControls等不能统一控制。 原因:模型有多个材质,在blender中合并的时候,…...
vue 本地自测iframe通讯
使用 postMessage API 来实现跨窗口(跨域)的消息传递。postMessage 允许你安全地发送消息到其他窗口,包括嵌套的 iframe,而不需要担心同源策略的问题。 发送消息(父应用) 1. 父应用:发送消息给…...
C++:单例模式
创建自己的对象,同时确保对象的唯一性。 单例类只能有一个实例☞静态成员static☞静态成员 必须类外初始化 单例类必须自己创建自己的唯一实例 单例类必须给所有其他对象提供这一实例 静态成员类内部可以访问 构造函数私有化☞构造函数私有外部不能创建&#x…...
SOME/IP 协议详解——信息格式
文章目录 1. 头部格式1.1 消息 ID(Message ID)1.2 长度(Length)1.3 请求 ID(Request ID)1.4 协议版本(Protocol Version):1.5 接口版本(Interface Version&am…...
C# GDI+数码管数字控件
调用方法 int zhi 15;private void button1_Click(object sender, EventArgs e){if (zhi > 19){zhi 0;}lcdDisplayControl1.DisplayText zhi.ToString();} 运行效果 控件代码 using System; using System.Collections.Generic; using System.Drawing.Drawing2D; using …...
在交叉编译中,常见的ELF(elf)到底是什么意思?
ELF 是 Executable and Linkable Format 的缩写,中文翻译为“可执行与可链接格式”。它是一种通用的文件格式,主要用于存储可执行文件、目标文件(编译后的中间文件)、动态库(.so 文件)以及内存转储文件&…...
Unity开发AR之Vuforia-MultiTarget笔记
前言 在增强现实(AR)技术蓬勃发展的今天,越来越多的开发者开始探索如何将AR应用于各种场景中。Vuforia作为一个领先的AR开发平台,为开发者提供了强大的工具和功能,使得创建AR体验变得更加简单和直观。本文将为您介绍Vuforia的基本概念、特点,以及如何配置和使用MultiTar…...
深入解析 Oracle 的聚合函数 ROLLUP
目录 深入解析 Oracle 的聚合函数 ROLLUP一、ROLLUP 函数概述二、ROLLUP 函数语法三、ROLLUP 实例详解(一)基础分组聚合(二)引入 ROLLUP 函数(三)ROLLUP 与 NULL 值(四)多列复杂分组…...
Wend看源码-Java-集合学习(List)
摘要 本篇文章深入探讨了基于JDK 21版本的Java.util包中提供的多样化集合类型。在Java中集合共分类为三种数据结构:List、Set和Queue。本文将详细阐述这些数据类型的各自实现,并按照线程安全性进行分类,分别介绍非线程安全与线程安全的实现方…...
【软件】教务系统成绩提交工具使用步骤
【软件】教务系统成绩提交工具使用步骤 零、快速开始 安装 与大多数软件一样,安装步骤很简单,一直点击“下一步”即可快速完成安装,安装完成后,在桌面会有一个软件图标,双击即可打开软件主界面。 导入成绩到Excel中…...
IPsec协议,网络安全的秘密
IPsec概述 IPsec是一组基于网络层的安全协议,是保护IP数据包在网络传输过程中保持安全、隐秘以及真实。通过对IP数据包进行一些加密、认证,来防止数据在传输过程中被窃取、篡改甚至伪造,IPsec在企业内部网络的通信、远程办公、云服务连接等场…...
浅谈下Spring MVC的执行流程
什么是Spring MVC Spring MVC是一个基于Java的Web框架,用于构建Web应用程序。 它是Spring Framework的一部分,它提供了模型-视图-控制器(MVC)架构。 支持RESTful风格的URL请求,易于与其他视图技术集成,如…...
khadas edge2安装ubuntu22.04与ubuntu20.04 docker镜像
khadas edge2安装ubuntu22.04与ubuntu20.04 docker镜像 一、资源准备1.1 镜像文件1.2 刷机工具1.3 ubuntu20.04 docker镜像(具备demon无人机所需各种驱动) 二、开始刷机(安装ubuntu22.04系统)2.1 进入刷机状态2.2 刷机 三、docker…...
关于iview组件中使用 table , 绑定序号分页后序号从1开始的解决方案
问题描述:iview使用table 中type: "index",分页之后 ,索引还是从1开始,试过绑定后台返回数据的id, 这种方法可行,就是后台返回数据的每个页面id都不完全是按照从1开始的升序,因此百度了下,找到了…...
【磁盘】每天掌握一个Linux命令 - iostat
目录 【磁盘】每天掌握一个Linux命令 - iostat工具概述安装方式核心功能基础用法进阶操作实战案例面试题场景生产场景 注意事项 【磁盘】每天掌握一个Linux命令 - iostat 工具概述 iostat(I/O Statistics)是Linux系统下用于监视系统输入输出设备和CPU使…...
全球首个30米分辨率湿地数据集(2000—2022)
数据简介 今天我们分享的数据是全球30米分辨率湿地数据集,包含8种湿地亚类,该数据以0.5X0.5的瓦片存储,我们整理了所有属于中国的瓦片名称与其对应省份,方便大家研究使用。 该数据集作为全球首个30米分辨率、覆盖2000–2022年时间…...
Springcloud:Eureka 高可用集群搭建实战(服务注册与发现的底层原理与避坑指南)
引言:为什么 Eureka 依然是存量系统的核心? 尽管 Nacos 等新注册中心崛起,但金融、电力等保守行业仍有大量系统运行在 Eureka 上。理解其高可用设计与自我保护机制,是保障分布式系统稳定的必修课。本文将手把手带你搭建生产级 Eur…...
CMake 从 GitHub 下载第三方库并使用
有时我们希望直接使用 GitHub 上的开源库,而不想手动下载、编译和安装。 可以利用 CMake 提供的 FetchContent 模块来实现自动下载、构建和链接第三方库。 FetchContent 命令官方文档✅ 示例代码 我们将以 fmt 这个流行的格式化库为例,演示如何: 使用 FetchContent 从 GitH…...
08. C#入门系列【类的基本概念】:开启编程世界的奇妙冒险
C#入门系列【类的基本概念】:开启编程世界的奇妙冒险 嘿,各位编程小白探险家!欢迎来到 C# 的奇幻大陆!今天咱们要深入探索这片大陆上至关重要的 “建筑”—— 类!别害怕,跟着我,保准让你轻松搞…...
负载均衡器》》LVS、Nginx、HAproxy 区别
虚拟主机 先4,后7...
渗透实战PortSwigger Labs指南:自定义标签XSS和SVG XSS利用
阻止除自定义标签之外的所有标签 先输入一些标签测试,说是全部标签都被禁了 除了自定义的 自定义<my-tag onmouseoveralert(xss)> <my-tag idx onfocusalert(document.cookie) tabindex1> onfocus 当元素获得焦点时(如通过点击或键盘导航&…...
Electron简介(附电子书学习资料)
一、什么是Electron? Electron 是一个由 GitHub 开发的 开源框架,允许开发者使用 Web技术(HTML、CSS、JavaScript) 构建跨平台的桌面应用程序(Windows、macOS、Linux)。它将 Chromium浏览器内核 和 Node.j…...
Linux【5】-----编译和烧写Linux系统镜像(RK3568)
参考:讯为 1、文件系统 不同的文件系统组成了:debian、ubuntu、buildroot、qt等系统 每个文件系统的uboot和kernel是一样的 2、源码目录介绍 目录 3、正式编译 编译脚本build.sh 帮助内容如下: Available options: uboot …...
