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

Spring Bean 为何“难产”?攻克构造器注入的依赖与歧义

本文已收录在Github,关注我,紧跟本系列专栏文章,咱们下篇再续!

  • 🚀 魔都架构师 | 全网30W技术追随者
  • 🔧 大厂分布式系统/数据中台实战专家
  • 🏆 主导交易系统百万级流量调优 & 车联网平台架构
  • 🧠 AIGC应用开发先行者 | 区块链落地实践者
  • 🌍 以技术驱动创新,我们的征途是改变世界!
  • 👉 实战干货:编程严选网

1 案例:定义的 Bean 缺少隐式依赖

有时把一个类定义成 Bean,又觉得这Bean定义除了加些 Spring 注解,好像平淡无奇!以致在后续使用时随心定义:

package com.javaedge.spring.service;@Service
public class MyService {private String myServiceName;public MyService(String myServiceName) {this.myServiceName = myServiceName;}public String sayHello() {return "hello Java";}
}

MyService显式定义了一个构造器。但上面代码不是永远都能正确运行,有时报错:

***********************************
APPLICATION FAILED TO START
***********************************Description:Parameter 0 of constructor in com.javaedge.spring.service.MyService required a bean of type 'java.lang.String' that could not be found.The injection point has the following annotations:
- @org.springframework.beans.factory.annotation.Autowired(required=true)Action:Consider defining a bean of type 'java.lang.String' in your configuration.

2 解惑

创建一个 Bean 时,调AbstractAutowireCapableBeanFactory的

2.1 createBeanInstance

  1. 寻找构造器
  2. 通过反射调用构造器创建实例
// Candidate constructors for autowiring?
Constructor<?>[] ctors = determineConstructorsFromBeanPostProcessors(beanClass, beanName);
if (ctors != null || mbd.getResolvedAutowireMode() == AUTOWIRE_CONSTRUCTOR ||mbd.hasConstructorArgumentValues() || !ObjectUtils.isEmpty(args)) {return autowireConstructor(beanName, mbd, ctors, args);
}

2.2 determineConstructorsFromBeanPostProcessors

Spring先执行它获取构造器:

仅一个有效实现类。然后通过 autowireConstructor 带着构造器去创建实例。

看ConstructorResolver的

2.3 instantiate

autowireConstructor方法要创建实例:

  • 不仅要知道是啥构造器
  • 还要知道构造器对应参数

从最后创建实例的方法名也可看出:

private Object instantiate(String beanName, RootBeanDefinition mbd, Constructor<?> constructorToUse, Object[] argsToUse) 

argsToUse咋获取?即当已知构造器ServiceImpl(String serviceName),要创建 ServiceImpl 实例,咋确定serviceName值?

使用 Spring,不能直接显式 new 创建实例。Spring只能寻找依赖作为构造器调用参数。那这参数咋获取?

看ConstructorResolver的

2.4 autowireConstructor

argsHolder = createArgumentArray(beanName, mbd, resolvedValues, bw, paramTypes, paramNames,getUserDeclaredConstructor(candidate), autowiring, candidates.length == 1);

可调用 createArgumentArray 构建调用构造器的参数数组,最终从 BeanFactory 获取 Bean:

即根据参数寻找对应 Bean。本案例中,如找不到对应 Bean,抛异常,提示装配失败。

3 修正

定义一个类为 Bean,若再显式定义构造器,则该 Bean 在构建时,会自动根据构造器参数定义寻找对应 Bean,反射创建该 Bean。可直接定义一个能让 Spring 装配给 MyService 构造器参数的 Bean,如:

package com.javaedge.spring.service;@Service
public class MyService {private String myServiceName;public String sayHello() {return "hello Java";}// 该bean装配给MyService的构造器参数-myServiceName@Beanpublic String myServiceName() {return "MyServiceName";}
}

综上,使用 Spring 时,别想当然认为定义的 Bean 也可在非 Spring 场合直接new!

若不精通 Spring 的隐式规则,在修正问题后,可能写出更多看起来好像可运行的程序:

@Service
public class MyService {private String myServiceName;public MyService(String myServiceName) {this.myServiceName = myServiceName;}public MyService(String myServiceName, String otherParam) {this.myServiceName = myServiceName;}

review这段代码,可能不会发现什么问题,毕竟 String 类型可自动装配,无非加个 String 参数。但若你知 Spring 内部用反射构建 Bean,就发现问题:存在两个构造器,都可调用时,到底调用哪个?最终 Spring 无从选择,只能尝试调用默认构造器,而默认构造器不存在:

所以报错:

http://www.xdnf.cn/news/10838.html

相关文章:

  • AI+在线教育系统源码:开发智能化互动网校平台全流程详解
  • 【相机基础知识与物体检测】更新中
  • 【北邮 操作系统】第十三章 I/O系统
  • 高考数学易错考点01 | 临阵磨枪
  • Spine工具入门教程4之网格与权重
  • SpringAI系列 - MCP篇(三) - MCP Client Boot Starter
  • 【C++高级主题】多重继承下的类作用域
  • 面向对象系统中对象交互的架构设计哲学
  • 集成学习之Bagging,Boosting,随机森林
  • Vue3 + Vite:我的 Qiankun 微前端主子应用实践指南
  • 杭州白塔岭画室怎么样?和燕壹画室哪个好?
  • LEAP模型
  • MongoDB-6.0.24 主从复制搭建和扩容缩容详解
  • Java垃圾回收机制深度解析:从理论到实践的全方位指南
  • 【Typst】4.导入、包含和读取
  • 【算法设计与分析】实验——汽车加油问题, 删数问题(算法实现:代码,测试用例,结果分析,算法思路分析,总结)
  • 行列式详解:从定义到应用
  • 滚动部署详解
  • Qt踩坑记录
  • 【Spring AI 1.0.0】Spring AI 1.0.0框架快速入门(1)——Chat Client API
  • 湖北理元理律师事务所:法律视角下的债务优化与生活平衡之道
  • 如何在 vue 中实现一个自定义拖拽的指令或插件
  • Spring @Autowired自动装配的实现机制
  • 手机邮箱APP操作
  • 风机巡检方案艰难之路
  • 全球高尔夫球场运营服务市场:现状、挑战与未来发展
  • Cursor使用最佳实践总结
  • 用 PyQt5 打造一个可视化 JSON 数据解析工具
  • 第二篇: 深入解析模型上下文协议(MCP):技术架构、核心组件与深远影响
  • SpringBoot2.3.1集成Knife4j接口文档