当前位置: 首页 > 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。

相关文章:

  • C#通过NTP服务器获取NTP时间
  • 鸿蒙版电影app设计开发
  • 鸿蒙NEXT开发正则工具类(ArkTs)
  • django admin 设置字段不可编辑
  • php安装swoole扩展
  • 【halcon】tuple_sort_index 和 select_obj 配合使用 详解
  • [特殊字符]实战:使用 Canal + MQ + ES + Redis + XXL-Job 打造高性能地理抢单系统
  • ASP.NET图片盗链防护指南
  • 总线位宽不变,有效数据位宽变化的缓存方案
  • 概率论与统计(不确定性分析)主要应用在什么方面?涉及到具体知识是什么?
  • 深入解析 npm 与 Yarn:Node.js 包管理工具对比与选型指南
  • 考研系列-计算机组成原理第五章、中央处理器
  • Spring Cloud Stream喂饭级教程【搜集全网资料整理】
  • 【Fifty Project - D18】
  • 【Flutter】Unity 三端封装方案:Android / iOS / Web
  • NGINX `ngx_http_core_module` 深度解读与实战指南
  • 晶晨S905L/LB芯片_安卓11.0_已适配移动遥控_支持外置网卡_支持IPV6_通刷线刷包
  • 通过ThreadLocal存储登录用户信息
  • rt-linux下的D状态的堆栈抓取及TASK_RTLOCK_WAIT状态
  • 使用 OpenCV 和 dlib 进行人脸检测
  • 吕国范任河南省人民政府副省长
  • 野猪穿过江苏电视台楼前广场,被抓捕后送往红山森林动物园
  • 学校食堂饭菜有蛆?举报人遭值班人员辱骂?四川苍溪县教育局回应
  • 关键词看中国经济“一季报”:稳,开局良好看信心
  • 天津外国语大学原校长修刚突发疾病去世,享年68岁
  • 起底网红热敷贴“苗古金贴”:“传承人”系AI生成,“千年秘方”实为贴牌货