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.pswp.cn/bicheng/83607.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

华为云Flexus+DeepSeek征文|实战体验云服务器单机部署和CCE高可用的架构AI赋能

前引&#xff1a;“在数字化浪潮汹涌澎湃的今天&#xff0c;企业对云计算服务的需求已从基础架构支撑&#xff0c;逐步转向更深层次的AI赋能与业务创新驱动。面对复杂多变的市场环境&#xff0c;选择一个强大、可靠且具备前瞻性的云服务伙伴&#xff0c;无疑是企业实现高速增长…

雷卯针对易百纳G610Q-IPC-38E 模组防雷防静电方案

一、应用场景 1、智能监控 2、智能家居 3、工业自动化 4、机器人 5、智能交通 6、医疗影像 7、教育科研 二、 功能概述 1 HI3516CV610&#xff08;ARM Cortex-A7 MP2&#xff09; 2 AI算力 1Tops 3 模组集成 4M30FPS Sensor&#xff0c;支持最高 6M30fps 的 ISP 图像…

生成对抗网络(GAN)基础原理深度解析:从直观理解到形式化表达

摘要 本文详细解析 生成对抗网络&#xff08;GAN&#xff09; 的 核心原理&#xff0c;从通俗类比入手&#xff0c;结合印假钞与警察博弈的案例阐述生成器 与 判别器 的对抗机制&#xff1b;通过模型结构示意图&#xff0c;解析 噪声采样、样本生成 及判别流程&#xff1b;基于…

OptiStruct结构分析与工程应用:无限元法介绍

13.3 无限元方法 本节将详细阐述如何利用无限元方法求解外声场分析&#xff0c;具体包括无限元方法基本理论&#xff0c;无限单元介绍、无限元分析建模指南及检查&#xff0c;最后以一个实例讲解整个分析设置过程。 13.3.1 无限元分析基础理论 无限元求解外声场的基本原理如…

判断:有那种使用了局部变量的递归过程在转换成非递归过程时才必须使用栈

这道题的关键在于理解递归转非递归与 “是否用栈” 的本质逻辑&#xff0c;和 “局部变量” 无关&#xff0c;核心看递归的调用上下文是否需要保存。 一、递归的本质&#xff1a;依赖 “调用栈” 递归函数执行时&#xff0c;系统会用调用栈保存&#xff1a; 每层递归的参数、…

leetcode1443. 收集树上所有苹果的最少时间-medium

1 题目&#xff1a;收集树上所有苹果的最少时间 官方标定难度&#xff1a;中 给你一棵有 n 个节点的无向树&#xff0c;节点编号为 0 到 n-1 &#xff0c;它们中有一些节点有苹果。通过树上的一条边&#xff0c;需要花费 1 秒钟。你从 节点 0 出发&#xff0c;请你返回最少需…

MySQL 索引底层原理剖析:B+ 树结构、索引创建维护与性能优化策略全解读

引言 在 MySQL 数据库的世界里&#xff0c;索引是提升查询性能的关键利器。然而&#xff0c;很多开发者虽然知道索引的重要性&#xff0c;但对于索引背后的底层原理却知之甚少。本文将深入 MySQL 索引的底层实现&#xff0c;剖析 B 树的结构特点&#xff0c;以及如何利用这些知…

【Delphi】实现在多显示器时指定程序运行在某个显示器上

在多显示器时代&#xff0c;经常会出现期望将程序运行在某个指定的显示器上&#xff0c;特别是在调试程序的时候&#xff0c;期望切换分辨率&#xff0c;单步调试时&#xff0c;此时容易导致互相卡住&#xff0c;非常不方便&#xff0c;但是通过指定程序运行在不同的显示器上就…

不动产登记区块链系统(Vue3 + Go + Gin + Hyperledger Fabric)

好久没有介绍过新项目的制作了&#xff0c;之前做的一直都是Fisco Bcos的项目&#xff0c;没有介绍过Hyperledger Fabric的项目&#xff0c;这次来给大家分享下。 系统概述 不动产登记与交易平台是一个基于Hyperledger Fabric的综合性管理系统&#xff0c;旨在实现不动产登记…

论文阅读笔记——Large Language Models Are Zero-Shot Fuzzers

TitanFuzz 论文 深度学习库&#xff08;TensorFlow 和 Pytorch&#xff09;中的 bug 对下游任务系统是重要的&#xff0c;保障安全性和有效性。在深度学习&#xff08;DL&#xff09;库的模糊测试领域&#xff0c;直接生成满足输入语言(例如 Python )语法/语义和张量计算的DL A…

cocos3.X的oops框架oops-plugin-excel-to-json改进兼容多表单导出功能

在使用oops框架的过程中&#xff0c;它的导出数据并生成数据结构的插件oops-plugin-excel-to-json有些小的坑点&#xff0c;为满足我个人习惯&#xff0c;对此部分进行了一个小的修改&#xff0c;有需要的拿去用&#xff0c;记录下供大家参考&#xff1b; 一、配置&#xff1a;…

解决IDE编译JAVA项目时出现的OOM异常问题

出现的异常如图&#xff1a; java.lang.0utOfMemoryError:Java heap space 解决方案&#xff1a; 文件 --> 设置 搜索 编译器&#xff08;就点击编译器这行&#xff09;&#xff0c;找到构建进程&#xff0c;共享堆大小&#xff0c;设置大一些&#xff0c;例如 2048 MB。 …

【Linux内核】设备模型之udev技术详解

目录 1. udev技术概述 2. 技术层次分析 2.1 内核层交互 2.2 规则引擎层 2.3 用户空间实现 3. 关键技术要点 3.1 动态设备节点管理 3.2 热插拔处理 3.3 模块化规则系统 3.3.1. 变量替换功能 3.3.2. 条件判断能力 3.3.3. 实现机制 3.3.4 应用场景 3.3.5 扩展能力 4…

群论在现代密码学中的应用探索与实践 —— 从理论到C语言实现

1. 引言&#xff1a;数字时代的信息安全挑战 随着互联网和数字技术的快速发展&#xff0c;信息安全问题变得日益严峻。无论是个人隐私保护&#xff0c;还是企业数据安全&#xff0c;乃至国家安全&#xff0c;都依赖于有效的加密技术保障信息的机密性和完整性。网络攻击、数据泄…

前端开发处理‘流式数据’与‘非流式数据’,在接收完整与非完整性数据时应该如何渲染和使用

在前端开发中&#xff0c;处理 非流式数据 和 流式数据 的方式不同。根据是否完整接收数据、是否实时渲染的需求&#xff0c;可以分为以下四种典型场景&#xff1a; 一、四类常见场景总结 类型数据完整性是否实时渲染适用技术/方法A完整数据&#xff08;一次性返回&#xff09…

thymeleaf直接调用Spring Bean中定义的方法

thymeleaf中可以使用表达式工具对象&#xff0c;通过符号直接调Spring Bean中定义的方法 Spring Bean Component public class InvokeMethodBean {public String fun() { return "fun";} }thymeleaf中调用 <div th:text"${invokeMethodBean.fun()}"&…

虚拟斯德哥尔摩症候群:用户为何为缺陷AI辩护?

当韩国用户美咲连续第七次为虚拟男友的算法错误辩解&#xff1a;“他只是太累了才会说伤人的话”&#xff0c;心理医生在诊断书上写下“数字依赖伴随认知失调”。这种现象并非孤例——斯坦福2024年研究显示&#xff0c;62%长期使用情感AI的用户会主动为系统缺陷寻找合理化解释&…

tryhackme——Abusing Windows Internals(进程注入)

文章目录 一、Abusing Processes二、进程镂空三、线程劫持四、DLL注入五、Memory Execution Alternatives 一、Abusing Processes 操作系统上运行的应用程序可以包含一个或多个进程&#xff0c;进程表示正在执行的程序。进程包含许多其他子组件&#xff0c;并且直接与内存或虚…

[蓝桥杯]密码脱落

密码脱落 题目描述 X 星球的考古学家发现了一批古代留下来的密码。 这些密码是由 A、B、C、D 四种植物的种子串成的序列。 仔细分析发现&#xff0c;这些密码串当初应该是前后对称的&#xff08;也就是我们说的镜像串&#xff09;。 由于年代久远&#xff0c;其中许多种子…

Python绘图库及图像类型

折线图&#xff08;plot&#xff09; 绘图库介绍 Python中绘制折线图的全面指南_python绘制折线图-CSDN博客https://blog.csdn.net/2301_81064905/article/details/139689644 核心作用说明趋势分析揭示数据随时间推移的上升/下降趋势、周期性波动或转折点变化对比在单一图表…