SpringBoot中@Conditional注解的介绍及实践

 更新时间:2025年03月25日 08:41:50   作者:拾荒的小海螺  
在 Spring Boot 中,@Conditional 注解用于实现 条件化 Bean 装配,本文将详细介绍 @Conditional 相关的注解,并结合实际应用示例讲解其使用方式,感兴趣的小伙伴可以了解下

1、简述

在 Spring Boot 中,@Conditional 注解用于实现 条件化 Bean 装配,即根据特定的条件来决定是否加载某个 Bean。它是 Spring 框架中的一个扩展机制,常用于实现模块化、可配置的组件加载。

本文将详细介绍 @Conditional 相关的注解,包括 @ConditionalOnClass、@ConditionalOnMissingBean、@ConditionalOnProperty 等,并结合实际应用示例讲解其使用方式。

2、@Conditional 注解概述

@Conditional 是 Spring 4 引入的条件装配注解,它可以根据外部环境或配置的状态,决定是否创建 Bean。

其核心接口是:

public interface Condition {
    boolean matches(ConditionContext context, AnnotatedTypeMetadata metadata);
}

任何实现 Condition 接口的类,都可以用于自定义条件判断。

@Configuration
public class MyConfig {

    @Bean
    @Conditional(MyCondition.class) 
    public MyService myService() {
        return new MyService();
    }
}

其中 MyCondition.class 需要实现 Condition 接口,并提供判断逻辑。

3、Spring Boot 内置 @Conditional 相关注解

Spring Boot 提供了一些常见的 @Conditional 注解,简化了条件判断的逻辑:

注解作用
@ConditionalOnClass类路径下存在某个类时,才加载该 Bean
@ConditionalOnMissingClass类路径下不存在某个类时,才加载该 Bean
@ConditionalOnBean当容器中存在指定 Bean 时,才加载当前 Bean
@ConditionalOnMissingBean当容器中不存在指定 Bean 时,才加载当前 Bean
@ConditionalOnProperty当指定的配置属性满足条件时,才加载当前 Bean
@ConditionalOnExpression当指定的 SpEL 表达式为 true 时,才加载当前 Bean
@ConditionalOnJava当 Java 版本符合要求时,才加载当前 Bean
@ConditionalOnWebApplication当应用是 Web 应用时,才加载当前 Bean
@ConditionalOnNotWebApplication当应用不是 Web 应用时,才加载当前 Bean

3.1 @ConditionalOnClass 使用示例(类路径检测)

我们希望在 Spring Boot 项目中,当类路径下存在 com.example.MyLibrary 这个类时,才注册 MyService 这个 Bean。

import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

@Configuration
public class MyAutoConfiguration {

    @Bean
    @ConditionalOnClass(name = "com.example.MyLibrary")
    public MyService myService() {
        return new MyService();
    }
}

解释:

  • 如果 com.example.MyLibrary 存在,则 MyService 会被创建。
  • 如果 com.example.MyLibrary 不存在,则 MyService 不会被加载。

3.2 @ConditionalOnMissingBean(Bean 缺失时加载)

如果用户没有手动定义 MyService,则提供一个默认实现。

import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

@Configuration
public class MyAutoConfiguration {

    @Bean
    @ConditionalOnMissingBean
    public MyService myService() {
        return new MyService("Default Implementation");
    }
}

解释:

  • 如果 Spring 容器中已经存在 MyService,则不会创建新的 Bean。
  • 只有当 MyService 不存在时,才会注册默认实现。

3.3 @ConditionalOnProperty(基于配置项条件加载 Bean)

我们希望 MyService 只有在 app.feature.enabled=true 时才被创建。

import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

@Configuration
public class MyAutoConfiguration {

    @Bean
    @ConditionalOnProperty(name = "app.feature.enabled", havingValue = "true", matchIfMissing = false)
    public MyService myService() {
        return new MyService();
    }
}

解释:

  • 如果 application.properties 中包含 app.feature.enabled=true,则 MyService 会被创建。
  • 如果 app.feature.enabled=false,或者该属性未定义,则 MyService 不会被加载。

在 application.properties 中启用:

app.feature.enabled=true

3.4 @ConditionalOnBean(存在特定 Bean 时才加载)

当 UserService 存在时,才创建 OrderService。

import org.springframework.boot.autoconfigure.condition.ConditionalOnBean;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

@Configuration
public class OrderServiceConfiguration {

    @Bean
    @ConditionalOnBean(UserService.class)
    public OrderService orderService() {
        return new OrderService();
    }
}

解释:

  • 如果 UserService 存在,则 OrderService 也会被创建。
  • 如果 UserService 不存在,则 OrderService 不会被加载。

3.5 @ConditionalOnExpression(基于 SpEL 表达式加载)

当 server.port 大于 8080 时,才创建 AdvancedService。

import org.springframework.boot.autoconfigure.condition.ConditionalOnExpression;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

@Configuration
public class AdvancedConfig {

    @Bean
    @ConditionalOnExpression("#{T(java.lang.Integer).parseInt('${server.port:8080}') > 8080}")
    public AdvancedService advancedService() {
        return new AdvancedService();
    }
}

解释:

server.port > 8080 时,AdvancedService 会被加载。

其他情况下,不会创建该 Bean。

在 application.properties 中配置:

server.port=9090

3.6 结合 @ConditionalOnClass 实现 Starter 组件的热拔插

我们要创建一个 Spring Boot Starter 组件,如果用户的 classpath 下存在 RedisTemplate,则自动加载 Redis 相关的 Bean。

步骤:

创建 Starter 组件

@Configuration
@ConditionalOnClass(RedisTemplate.class)
public class RedisAutoConfiguration {

    @Bean
    @ConditionalOnMissingBean
    public RedisTemplate<String, Object> redisTemplate(RedisConnectionFactory factory) {
        RedisTemplate<String, Object> template = new RedisTemplate<>();
        template.setConnectionFactory(factory);
        return template;
    }
}

应用使用 Starter

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>

如果引入 Redis 依赖,RedisAutoConfiguration 会自动生效。

如果不引入 Redis 依赖,则 RedisTemplate 不会被创建。

4、总结

  • @Conditional 及其衍生注解使 Spring Boot 具备了自动配置和热拔插的能力。
  • @ConditionalOnClass 可用于判断某个类是否存在,常用于 Starter 组件的自动装配。
  • @ConditionalOnProperty 适用于基于配置的条件加载,增强灵活性。
  • @ConditionalOnBean 和 @ConditionalOnMissingBean 适用于组件依赖管理。

合理使用这些注解,可以构建更加模块化、灵活、可配置的 Spring Boot 应用。

以上就是SpringBoot中@Conditional注解的介绍及实践的详细内容,更多关于SpringBoot @Conditional注解的资料请关注脚本之家其它相关文章!

相关文章

  • 浅析SpringBoot如何解决CORS问题

    浅析SpringBoot如何解决CORS问题

    在前后端分离的开发模式中,经常会遇到 跨域资源共享(CORS) 的问题,本文将全面介绍 Spring Boot 中处理 CORS 的常见方法,大家可以根据需要进行选择
    2025-05-05
  • SpringBoot中实现启动任务的实现步骤

    SpringBoot中实现启动任务的实现步骤

    这篇文章主要介绍了SpringBoot中实现启动任务的实现步骤,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-09-09
  • java LRU(Least Recently Used )详解及实例代码

    java LRU(Least Recently Used )详解及实例代码

    这篇文章主要介绍了java LRU(Least Recently Used )详解及实例代码的相关资料,Java里面实现LRU缓存通常有两种选择,一种是使用LinkedHashMap,一种是自己设计数据结构,使用链表+HashMap,需要的朋友可以参考下
    2016-11-11
  • Java 重写与重载方法与区别详解

    Java 重写与重载方法与区别详解

    本篇文章通过实例详细介绍了重写与重载,以及他们的区别,需要的朋友可以参考下
    2017-04-04
  • spring-boot https证书双向认证配置的实现

    spring-boot https证书双向认证配置的实现

    本文详细介绍SpringBoot项目中配置自签名CA证书、签发服务端和客户端证书,生成PKCS12格式的证书,及设置SSL/TLS配置,实现双向认证,具有一定的参考价值,感兴趣的可以了解一下
    2025-08-08
  • Java实现stream的三个常用方式(toMap,groupingBy,findFirst)

    Java实现stream的三个常用方式(toMap,groupingBy,findFirst)

    本文主要介绍了Java实现stream的三个常用方式,主要包括toMap,groupingBy,findFirst,具有一定的参考价值,感兴趣的可以了解一下
    2023-10-10
  • Java多线程Future松获取异步任务结果轻松实现

    Java多线程Future松获取异步任务结果轻松实现

    这篇文章主要为大家介绍了Java多线程Future松获取异步任务结果轻松实现方法,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-04-04
  • 带你用Java全面剖析类和对象

    带你用Java全面剖析类和对象

    下面小编就为大家带来一篇全面理解Java类和对象。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2021-09-09
  • 详解Spring关于@Resource注入为null解决办法

    详解Spring关于@Resource注入为null解决办法

    这篇文章主要介绍了详解Spring关于@Resource注入为null解决办法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-05-05
  • Java多线程事务回滚@Transactional失效处理方案

    Java多线程事务回滚@Transactional失效处理方案

    这篇文章主要介绍了Java多线程事务回滚@Transactional失效处理方案,文章围绕主题展开详细的内容介绍,具有一定的参考价值,需要的朋友可以参考一下
    2022-08-08

最新评论