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

http://www.mrgr.cn/news/100471.html

相关文章:

  • CI/CD自动化部署(持续集成和持续交付/部署)
  • Linux常用命令23——usermod修改用户信息
  • 《全球反空间能力》报告翻译——部分1
  • Vue3:component(组件:uniapp版本)
  • 第一个 servlet请求
  • K8S Pod 常见数据存储方案
  • Java SE(3)——程序逻辑控制,输入输出
  • MySQL----查询
  • 数据结构二叉树与二叉搜索树c实现代码
  • 使用Open Compass进行模型评估,完成AI模型选择
  • PTA -L1-005 考试座位号(BufferedReader、Arraylist动态数组、Map)
  • 数据结构强化篇
  • 【文心快码】确实有点东西!
  • 【Maven】特殊pom.xml配置文件 - BOM
  • uniapp: 低功耗蓝牙(BLE)的使用
  • 前端Vue项目处理跨域请求问题解决方案(后端未加cors),前端调后端
  • Day23-Web开发——Linux
  • Java安全之cc链学习集合
  • Win11 配置 Git 绑定 Github 账号的方法与问题汇总
  • 【Spring Boot】Maven中引入 springboot 相关依赖的方式