优雅终结启动顺序噩梦:ObjectProvider —— Spring 4.3 开始引入

优雅终结启动顺序噩梦:ObjectProvider —— Spring 4.3 开始引入
🧑 博主简介ZEEKLOG博客专家「历代文学网」(PC端可以访问:https://lidaiwenxue.com/#/?__c=1000,移动端可关注公众号 “ 心海云图 ” 微信小程序搜索“历代文学”)总架构师,首席架构师,也是联合创始人!16年工作经验,精通Java编程高并发设计分布式系统架构设计Springboot和微服务,熟悉LinuxESXI虚拟化以及云原生Docker和K8s,热衷于探索科技的边界,并将理论知识转化为实际应用。保持对新技术的好奇心,乐于分享所学,希望通过我的实践经历和见解,启发他人的创新思维。在这里,我希望能与志同道合的朋友交流探讨,共同进步,一起在技术的世界里不断学习成长。
🤝商务合作:请搜索或扫码关注微信公众号 “ 心海云图


在这里插入图片描述

优雅终结启动顺序噩梦:ObjectProvider —— Spring 4.3 开始引入

从“饥渴式依赖”到“按需获取”,一次依赖注入的思想跃迁

缘起:一个再普通不过的配置类,为何启动就报错?

就在上周,我在维护一个 Spring Boot 4.0 项目时遇到了一个棘手的问题:

@ConfigurationpublicclassWebmvcEncryptionConfiguration{@AutowiredprivateHttpMessageConverter<Object> httpMessageConverter;// 注入总为 null@BeanFilterRegistrationBean<EncryptedFilter>encryptedFilterRegistrationBean(){// 启动时抛出 NullPointerException,httpMessageConverter 未初始化!returnnewFilterRegistrationBean<>(newEncryptedFilter(httpMessageConverter));}}

无论我怎么调整注入方式——字段注入、构造器注入、方法参数注入——HttpMessageConverter 始终是 null。更诡异的是,只要注释掉这个配置类,整个应用就能正常启动,HttpMessageConverter 也能被完美初始化

直觉告诉我:这不是注入“写错了”,而是 FilterRegistrationBean 的初始化时机早于 HttpMessageConverter 的自动配置

Spring 容器尚未完成 WebMvc 基础设施的装配,我的过滤器却已经急不可耐地索要依赖——这就像在清晨 5 点去咖啡馆要一杯现磨手冲,咖啡师还在通勤路上。

如何让依赖“等等再给”?

答案就藏在 Spring 4.3 引入的一个低调接口中:ObjectProvider


一、版本之问:ObjectProvider 究竟出生在哪一年?

关于 ObjectProvider 的引入版本,网上存在不少混淆信息。我需要在这里正本清源:

Spring Framework 4.3 首次引入 ObjectProvider
Spring 5.0 说、Spring Boot 2.0 说 均为误传

权威证据链

  1. Spring 官方博客(2016年3月)明确写道:“Spring Framework 4.3 引入了 ObjectProvider,它是现有 ObjectFactory 接口的扩展,提供 getIfAvailablegetIfUnique 等便捷签名”
  2. 版本时间线:Spring 4.3.0.RC1 发布于 2016年3月,4.3.0.GA 发布于 2016年5月;而 Spring 5.0 在 2017年9月才正式发布
  3. 历史实证:有开发者反馈在 Spring 4.2.4 中遇到 NoClassDefFoundError: org/springframework/beans/factory/ObjectProvider,升级到 4.3 后解决

为什么会有 5.0 的说法?因为 Spring 5.0 和 Spring Boot 2.0 增强了ObjectProvider(如添加 orderedStream() 方法),但它真正的诞生时刻是 2016 年 5 月,Spring Framework 4.3 GA


二、原理深潜:ObjectProvider 为什么能解决顺序问题?

2.1 两种依赖获取哲学的较量

要理解 ObjectProvider 的优雅,首先要看清 @Autowired 的本质:

维度@AutowiredObjectProvider<T>
获取时机Bean 实例化立即解析调用 getObject()延迟解析
依赖强度默认强依赖(required=true可选依赖,允许不存在
多实例处理必须配合 @Qualifier/@Primary支持运行时动态筛选、流式处理
原型 Bean注入固定实例(违背原型语义)每次调用 getObject() 获取新实例
异常处理启动即失败将异常推迟到业务运行时

形象的比喻

  • @Autowired“咖啡必须在我进办公室前就放在桌上”
  • ObjectProvider“给我一张咖啡券,我想喝的时候自己去打”

2.2 Spring 源码级的特殊对待

为什么 ObjectProvider 能“躲过”启动时的依赖解析?秘密藏在 DefaultListableBeanFactory.resolveDependency() 中 :

@OverridepublicObjectresolveDependency(DependencyDescriptor descriptor,...){// 1. Optional<T>if(Optional.class== descriptor.getDependencyType()){returncreateOptionalDependency(descriptor, requestingBeanName);}// 2. ObjectFactory<T>、ObjectProvider<T> —— 重点在这里!elseif(ObjectFactory.class== descriptor.getDependencyType()||ObjectProvider.class== descriptor.getDependencyType()){// 不触发真正的依赖解析,直接返回一个“懒加载代理”returnnewDependencyObjectProvider(descriptor, requestingBeanName);}// ... 其他情况else{returndoResolveDependency(descriptor,...);// 立即解析}}

关键结论:当 Spring 发现你注入的是 ObjectProvider<T> 时,它不会去容器中查找 T 类型的 Bean,而是直接给你一个 DependencyObjectProvider 对象。真正的 Bean 查找被推迟到你第一次调用 getObject()getIfAvailable() 的时刻。

这正是解决 FilterRegistrationBean 初始化顺序问题的终极武器——我的 Filter 可以提前注册,但 HttpMessageConverter 可以“按需再取”


三、实战改造:从“饥渴注入”到“按需获取”

3.1 问题代码的完整解决方案

publicclassEncryptedFilterimplementsFilter{privatefinalWebClientConfigProperties properties;privatefinalRequestMappingHandlerMapping handlerMapping;privatefinalObjectProvider<HttpMessageConverter<?>> converterProvider;// 注入提供者publicEncryptedFilter(WebClientConfigProperties properties,RequestMappingHandlerMapping handlerMapping,ObjectProvider<HttpMessageConverter<?>> converterProvider){this.properties = properties;this.handlerMapping = handlerMapping;this.converterProvider = converterProvider;}@OverridepublicvoiddoFilter(ServletRequest request,ServletResponse response,FilterChain chain)throwsIOException,ServletException{// 真正需要转换器时才去获取——此时容器已完成自动配置HttpMessageConverter<?> converter = converterProvider.getIfAvailable();if(converter !=null){// 使用转换器处理加解密逻辑} chain.doFilter(request, response);}}

配置类(彻底抛弃字段注入):

@ConfigurationpublicclassWebmvcEncryptionConfiguration{@BeanFilterRegistrationBean<EncryptedFilter>encryptedFilterRegistrationBean(WebClientConfigProperties webClientConfigProperties,RequestMappingHandlerMapping requestMappingHandlerMapping,ObjectProvider<HttpMessageConverter<?>> httpMessageConverterProvider){// 参数注入EncryptedFilter filter =newEncryptedFilter( webClientConfigProperties, requestMappingHandlerMapping, httpMessageConverterProvider);// 传递的是 Provider,不是具体的 ConverterFilterRegistrationBean<EncryptedFilter> registrationBean =newFilterRegistrationBean<>(); registrationBean.setOrder(Ordered.HIGHEST_PRECEDENCE); registrationBean.setFilter(filter); registrationBean.setName("encryptedFilter"); registrationBean.addUrlPatterns("/*");return registrationBean;}}

改动极小,但彻底解决了启动顺序问题


四、ObjectProvider 的四大黄金场景

如果说解决启动顺序只是 ObjectProvider 的“意外之喜”,那么下面四个场景才是它真正的设计目标,也是每一位 Spring 开发者都应该掌握的进阶技巧。

场景一:优雅处理可选依赖(替代 @Autowired(required=false)

传统写法——丑陋的空值检查:

@ServicepublicclassNotificationService{@Autowired(required =false)privateSmsService smsService;// 可能为 nullpublicvoidsendAlert(String message){if(smsService !=null){// 每次都要判空 smsService.send(message);}}}

ObjectProvider 写法——函数式、零判空:

@ServicepublicclassNotificationService{@AutowiredprivateObjectProvider<SmsService> smsServiceProvider;publicvoidsendAlert(String message){// 存在才执行,不存在什么都不发生 smsServiceProvider.ifAvailable(sms -> sms.send(message));// 或者提供默认实现SmsService sms = smsServiceProvider.getIfAvailable(()->newMockSmsService());}}

场景二:在单例 Bean 中正确获取原型 Bean(最核心用途)

错误示范——90% 开发者踩过的坑 :

@Component@Scope("prototype")publicclassTaskProcessor{/* ... */}@Service// 默认单例publicclassTaskService{@AutowiredprivateTaskProcessor taskProcessor;// ❌ 仅在启动时注入一次,之后永远是同一个实例!publicvoidexecute(){ taskProcessor.process();// 每次用的都是同一个对象}}

ObjectProvider 拯救

@ServicepublicclassTaskService{@AutowiredprivateObjectProvider<TaskProcessor> taskProcessorProvider;publicvoidexecute(){// 每次调用都从容器获取全新的原型实例TaskProcessor processor = taskProcessorProvider.getObject(); processor.process();}}

与 @Lookup 对比

方式优点缺点
@Lookup注解简洁,无需显式注入仅能用于方法,调试困难
ObjectProvider灵活、可编程、支持条件判断需注入 Provider 对象
ApplicationContext功能最全耦合容器,性能稍差

结论原型 Bean 获取,首选 ObjectProvider

场景三:动态筛选多个同类型 Bean

当一个接口有多个实现时,传统方式必须指定 @Qualifier@Primary编译时就决定了用哪一个

ObjectProvider 实现运行时动态选择

publicinterfacePaymentService{booleansupports(String type);voidpay(Order order);}@ServicepublicclassPaymentProcessor{@AutowiredprivateObjectProvider<PaymentService> paymentServiceProvider;publicvoidprocessPayment(Order order,String paymentType){PaymentService service = paymentServiceProvider.stream().filter(ps -> ps.supports(paymentType)).findFirst().orElseThrow(()->newUnsupportedOperationException("不支持的支付方式")); service.pay(order);}}

加上排序支持(Spring 5.1+):

// 按照 @Order 或 Ordered 接口的顺序获取所有实现 paymentServiceProvider.orderedStream().forEach(PaymentService::someCommonOperation);

场景四:解决循环依赖

虽然 Spring 三级缓存能解决大部分 setter 注入的循环依赖,但对于构造器注入的循环依赖依然束手无策。ObjectProvider 可以轻松破局:

@ComponentpublicclassServiceA{privatefinalObjectProvider<ServiceB> serviceBProvider;publicServiceA(ObjectProvider<ServiceB> serviceBProvider){this.serviceBProvider = serviceBProvider;}publicvoiddoSomething(){// 需要 B 的时候再去拿,此时 B 一定已初始化完毕ServiceB b = serviceBProvider.getObject();}}@ComponentpublicclassServiceB{privatefinalServiceA serviceA;// 正常构造器注入publicServiceB(ServiceA serviceA){this.serviceA = serviceA;}}

五、与其他方案的对比:我该如何选择?

场景推荐方案理由
普通依赖注入@Autowired 构造器注入最简洁、最符合 DI 原则
依赖可能不存在ObjectProvider.getIfAvailable()无需判空、函数式风格
单例中获取原型 BeanObjectProvider最均衡:灵活、低耦合、性能优
运行时多实现动态选择ObjectProvider.stream()原生支持流式处理
仅需简单原型获取@Lookup代码最少,无额外注入
需要完全控制容器ApplicationContext功能最全,但耦合度高
启动顺序问题ObjectProvider唯一的“延迟查找”解

核心决策标准:只要存在 “依赖的创建/解析时机晚于当前 Bean 的初始化时机”,就应该考虑 ObjectProvider


六、结语:为什么说 ObjectProvider 是一颗“时间胶囊”?

回顾开篇的问题,FilterRegistrationBean 之所以无法直接注入 HttpMessageConverter,本质上是 基础设施组件与业务组件初始化阶段的不匹配

ObjectProvider 的伟大之处,不是它提供了什么复杂的功能,而是它改变了我们对依赖注入的思考方式

  • 从“饥渴式”到“按需式”:依赖不一定要在注入点就绪,可以等到真正使用时才获取
  • 从“静态绑定”到“动态解析”:依赖的选择可以从编译期推迟到运行期
  • 从“强依赖”到“可选依赖”:允许依赖的不确定性,并在语言层面优雅处理

这让我想起计算机科学中的一句名言:“计算机科学中的所有问题,都可以通过增加一个间接层来解决”ObjectProvider<T> 正是这样一层优雅的“间接”——它将“依赖”封装成“获取依赖的能力”,将“对象”升级为“对象提供者”。

当你下次在 Spring Boot 启动过程中遇到类似的顺序问题时,不妨问问自己:

“我是否可以在当前 Bean 中,不为具体的依赖,而为它的‘提供者’预留一个位置?”

这个答案,早在 2016 年春天,Spring 4.3 就已经为你准备好了。


附录:ObjectProvider 核心 API 速查表

方法行为典型场景
T getObject()获取 Bean,不存在/不唯一则抛异常必需依赖、原型 Bean
T getIfAvailable()存在则返回,否则返回 null可选依赖
T getIfAvailable(Supplier<T>)不存在则返回默认值提供降级方案
void ifAvailable(Consumer<T>)存在时执行操作函数式风格、零判空
T getIfUnique()唯一则返回,否则返回 null期望单一候选
Stream<T> stream()返回所有匹配 Bean 的流多实现处理
Stream<T> orderedStream()按 @Order 排序的流有序多实现
本文由真实生产问题驱动,结合 Spring 源码与官方文档撰写。

Read more

Flutter 三方库 arcane_helper_utils 的鸿蒙化适配指南 - 实现具备通用逻辑增强与多维开发脚手架的实用工具集、支持端侧业务开发的效率倍增实战

Flutter 三方库 arcane_helper_utils 的鸿蒙化适配指南 - 实现具备通用逻辑增强与多维开发脚手架的实用工具集、支持端侧业务开发的效率倍增实战

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.ZEEKLOG.net Flutter 三方库 arcane_helper_utils 的鸿蒙化适配指南 - 实现具备通用逻辑增强与多维开发脚手架的实用工具集、支持端侧业务开发的效率倍增实战 前言 在进行 Flutter for OpenHarmony 开发时,如何快速处理常见的字符串格式化、色值转换、日期计算或布尔值增强?虽然每一个功能都很小,但如果每个项目都重复造轮子,开发效率将大打折扣。arcane_helper_utils 是一款专注于极致实用的“瑞士军刀”型工具集。本文将探讨如何在鸿蒙端通过这类高内聚的 Utility 集实现极致、丝滑的业务交付。 一、原直观解析 / 概念介绍 1.1 基础原理 该库通过对 Dart 原生类型(Object, String, List, Map, Bool)

By Ne0inhk
宇树VR遥操与IL——从遥操程序xr_teleoperate到unitree_IL_lerobot:如何基于G1进行manipulation开发

宇树VR遥操与IL——从遥操程序xr_teleoperate到unitree_IL_lerobot:如何基于G1进行manipulation开发

前言 如之前的文章所述,我司「七月在线」正在并行开发多个订单,目前正在全力做好每一个订单,因为保密协议的原因,暂时没法拿出太多细节出来分享 但可以持续解读我们所创新改造或二次开发的对象,即解读paper和开源库「当然 有些paper/库还没开始用,但也可以提前解读,作为关注了解」 而对于我司人形开发的订单,截止到25年4月,背后的机器人多半基于这几家:宇树、智元、傅利叶、乐聚「之所以用的这几家,一半因为我和这些公司熟,一半因为客户已有其中某一家或某几家的本体 需在其基础上做定制开发,如其它厂商看到 有兴趣合作,欢迎私我,比如星动纪元、星海图、众擎等等」 * 通过此文《Fourier-Lerobot——把斯坦福人形动作策略iDP3封装进了Lerobot(含我司七月的idp3落地实践)》可知,傅利叶 把idp3 装进了lerobot * 类似的,宇树 通过此开源库「unitree_IL_lerobot」,也把lerobot 集成了下 该库包含了π0策略 且无论咱们是用傅利叶集成的lerobot—

By Ne0inhk
手把手教你用 OpenClaw + 飞书,打造专属 AI 机器人

手把手教你用 OpenClaw + 飞书,打造专属 AI 机器人

手把手教你用 OpenClaw + 飞书,打造专属 AI 机器人 当前版本 OpenClaw(2026.2.22-2)已内置飞书插件,无需额外安装。 你有没有想过,在飞书里直接跟 AI 对话,就像跟同事聊天一样自然? 今天这篇文章,带你从零开始,用 OpenClaw 搭建一个飞书 AI 机器人。全程命令行操作,10 分钟搞定。 一、准备工作 1.1 安装 Node.js(版本 ≥ 22) OpenClaw 依赖 Node.js 运行,首先确保你的 Node 版本不低于 22。 推荐使用 nvm 管理 Node

By Ne0inhk

【无人机3D路径规划】基于改进蝙蝠优化算法的无人机3D路径规划研究附Matlab代码

✅作者简介:热爱科研的Matlab仿真开发者,擅长毕业设计辅导、数学建模、数据处理、建模仿真、程序设计、完整代码获取、论文复现及科研仿真。 🍎 往期回顾关注个人主页:Matlab科研工作室  👇 关注我领取海量matlab电子书和数学建模资料  🍊个人信条:格物致知,完整Matlab代码获取及仿真咨询内容私信。 🔥 内容介绍  一、引言 在当今科技飞速发展的时代,无人机在众多领域得到了广泛应用,从物流配送、农业监测到航空测绘等。在这些应用场景中,无人机需要在三维空间中规划出一条安全、高效的飞行路径,以完成各种任务。传统的路径规划算法在处理复杂的 3D 环境时,往往存在收敛速度慢、易陷入局部最优等问题。蝙蝠优化算法(Bat Algorithm,BA)作为一种新兴的智能优化算法,模拟了蝙蝠的回声定位行为,为解决此类问题提供了新的思路。然而,标准的蝙蝠优化算法也有其局限性,因此本文聚焦于基于改进蝙蝠优化算法的无人机 3D 路径规划研究,旨在提升路径规划的性能。 二、蝙蝠优化算法基础 1. 蝙蝠回声定位模拟:蝙蝠在飞行过程中通过发出超声波,并根据回声来感知周围环

By Ne0inhk