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

Spring框架的ObjectProvider用法-笔记

在Spring框架中,ObjectProvider 是一个用于灵活获取Bean实例的接口,它允许开发者以编程方式有条件地或可选地获取Bean,而无需强制依赖注入,避免在Bean不存在时启动失败。

1. ObjectProvider 的核心功能

ObjectProvider 是Spring 5.0引入的接口,用于替代旧版的 ObjectFactory 和 FactoryBean,提供更灵活的Bean获取方式。其核心功能包括:

  • 可选依赖:允许获取Bean时不强制要求Bean存在
  • 处理多个Bean:当有多个匹配的Bean时,支持选择性获取(如唯一Bean、按类型/条件获取)。
  • 延迟初始化:按需获取Bean实例,避免提前初始化。

示例代码

@Configuration
public class UserAutoConfiguration {private UserFacadeClient userFacadeClient;public UserAutoConfiguration(ObjectProvider<UserFacadeClient> userFacadeClient) {this.userFacadeClient = userFacadeClient.getIfAvailable();}
}

代码关键点分析

  • 构造函数注入 ObjectProvider<UserFacadeClient>
    • Spring会自动注入一个 ObjectProvider<UserFacadeClient> 实例,该实例可以获取所有匹配的 UserFacadeClient Bean。
  • getIfAvailable() 方法
    • 该方法尝试获取当前可用的 UserFacadeClient Bean。如果存在至少一个Bean,则返回第一个匹配的实例;如果不存在,则返回 null不会抛出异常
    • 适合场景:当 UserFacadeClient 是一个可选依赖时(例如,某些环境下可能不需要该客户端),使用 ObjectProvider 可以避免Spring启动失败。

2.ObjectProvider 的其他常用方法

(1) getIfAvailable()
  • 作用:获取第一个可用的Bean实例,若不存在则返回 null
  • 示例
public UserAutoConfiguration(ObjectProvider<UserFacadeClient> userFacadeClient) {UserFacadeClient client = userFacadeClient.getIfAvailable();if (client != null) {// 使用客户端进行配置}
}
(2) getIfUnique()
  • 作用:获取唯一匹配的Bean实例,若存在多个Bean则抛出异常。
  • 示例
public UserAutoConfiguration(ObjectProvider<UserFacadeClient> userFacadeClient) {UserFacadeClient client = userFacadeClient.getIfUnique();// 如果存在多个Bean,会抛出 NoUniqueBeanDefinitionException
}
(3) getObject()
  • 作用:获取Bean实例,若存在多个Bean则抛出异常,若不存在则抛出 NoSuchBeanDefinitionException
  • 示例
public UserAutoConfiguration(ObjectProvider<UserFacadeClient> userFacadeClient) {//强制要求存在且唯一UserFacadeClient client = userFacadeClient.getObject();
}
(4) stream()
  • 作用:返回所有匹配的Bean的流(Stream),支持遍历多个Bean。
  • 示例
public UserAutoConfiguration(ObjectProvider<UserFacadeClient> userFacadeClient) {userFacadeClient.stream().forEach(client -> {// 处理所有匹配的Bean});
}
(5) orderedStream()
  • 作用:按Spring的排序规则(如 @Order 注解)返回Bean的流。
  • 示例
public UserAutoConfiguration(ObjectProvider<UserFacadeClient> userFacadeClient) {userFacadeClient.orderedStream().findFirst().ifPresent(client -> {// 使用排序后的第一个Bean});
}
(6) 总结
方法功能适用场景
getIfAvailable()获取第一个可用Bean,不存在则返回null可选依赖,避免强制要求Bean存在
getIfUnique()获取唯一Bean,否则抛异常需要唯一Bean时
getObject()获取Bean,强制要求存在且唯一强制依赖
stream()获取所有匹配Bean的流处理多个Bean

通过 ObjectProvider,开发者可以更灵活地控制Bean的获取逻辑,尤其在需要条件化配置或处理可选依赖时,能显著提升代码的健壮性和可维护性。

3. 典型使用场景示例

场景1:可选依赖

假设 UserFacadeClient 是一个可选的第三方客户端,某些环境中可能不需要:

@Configuration
public class UserAutoConfiguration {private final UserFacadeClient userFacadeClient;public UserAutoConfiguration(ObjectProvider<UserFacadeClient> provider) {this.userFacadeClient = provider.getIfAvailable();}@Beanpublic UserService userService() {if (userFacadeClient != null) {return new UserServiceWithUserFacade(userFacadeClient);} else {return new DefaultUserService();}}
}
场景2:处理多个Bean

如果有多个 UserFacadeClient 实现(如不同环境的客户端):

@Configuration
public class UserAutoConfiguration {public UserAutoConfiguration(ObjectProvider<UserFacadeClient> provider) {provider.stream().forEach(client -> {// 遍历所有客户端并进行配置});}
}

4. 对比传统 @Autowired 方式

传统方式(强制依赖)

@Configuration
public class UserAutoConfiguration {@Autowiredprivate UserFacadeClient userFacadeClient;// 如果没有UserFacadeClient Bean,Spring启动会失败
}

使用 ObjectProvider(可选依赖)

@Configuration
public class MessageAutoConfiguration {private UserFacadeClient userFacadeClient;public MessageAutoConfiguration(ObjectProvider<UserFacadeClient> provider) {// 有没有UserFacadeClient Bean,都不会影响Spring启动this.userFacadeClient = provider.getIfAvailable();}
}

区别

  • 强制性:传统方式要求Bean必须存在,否则启动失败;ObjectProvider 可以优雅处理不存在的情况。
  • 灵活性ObjectProvider 支持获取多个Bean或按条件选择,而传统方式只能获取单个Bean。

相关文章:

  • STM32F103_HAL库+寄存器学习笔记21 - CAN接收过滤器:CPU减负神器,提升系统效率的第一道防线
  • Java——封装(面向对象)
  • 迅雷精简绿色融合版【高速下载版】12.1.9.2870【11.2.2.1716】【20250426】
  • 检查 NetCDF Fortran的版本
  • 【Java-Day 2】Java开发利器:IntelliJ IDEA入门教程(安装、配置、项目创建、调试)
  • 零基础 学习Linux shell编程语法
  • 机器学习 | 基于回归模型的交通需求预测案例分析及代码示例
  • 【工具】scMultiMap基于单细胞多模态数据实现增强子与靶基因的细胞类型特异性映射
  • 基于STM32、HAL库的TSC2046IPWR触摸屏控制器驱动程序设计
  • 【同局域网/内网环境 Windows 远程桌面连接】
  • 动手学深度学习11.10. Adam算法-笔记练习(PyTorch)
  • Pacman-Multi-Agent Search
  • 深度解析:具身AI机器人领域最全资源指南(含人形机器人,多足机器人,灵巧手等精选资源)
  • AiCube 试用 - ADC 水位监测系统
  • 如何培养团队的责任感与归属感
  • 多节点同步协同电磁频谱监测任务分配方法简要介绍
  • 【vue3】购物车实战:从状态管理到用户体验的全流程实现
  • 测量电机的电阻、电感、磁链常数和极对数办法
  • Go语言之路————指针、结构体、方法
  • Python 基础核心知识
  • 利物浦提前四轮英超夺冠,顶级联赛冠军数追平曼联
  • 2025厦门体育产业采风活动圆满举行
  • 规范涉企案件审判执行工作,最高法今天发布通知
  • 人民日报:广东全力推动外贸稳量提质
  • 甘肃省原副省长赵金云被开除公职,甘肃省委表态:坚决拥护党中央决定
  • 恒瑞医药一季度营收72亿元,净利增超36%:授权交易推动利润增长