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
繁體地址,請注明出處:http://hk.pswp.cn/bicheng/83607.shtml
英文地址,請注明出處:http://en.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 核心作用說明趨勢分析揭示數據隨時間推移的上升/下降趨勢、周期性波動或轉折點變化對比在單一圖表…