Spring 构造器注入和setter注入的比较
一、比较说明
在 Spring 框架中,构造器注入(Constructor Injection)和 Setter 注入(Setter Injection)是实现依赖注入(DI)的两种主要方式。它们的核心区别在于依赖注入的时机、代码设计理念以及适用场景。以下是两者的详细比较:
1. 核心区别
| 特性 | 构造器注入 | Setter 注入 |
|---|---|---|
| 注入方式 | 通过类的构造方法注入依赖。 | 通过 Setter 方法注入依赖。 |
| 依赖不可变性 | 依赖通常声明为 final,确保对象创建后不可变。 | 依赖可变,可在对象生命周期中修改。 |
| 依赖必要性 | 强制要求依赖,适用于必需依赖。 | 可选依赖,允许部分依赖为 null。 |
| 初始化完整性 | 对象创建时即完成依赖注入,保证完全初始化。 | 对象可能处于“部分初始化”状态(依赖未完全注入)。 |
| 循环依赖处理 | 无法解决构造器级别的循环依赖(Spring 会抛出异常)。 | 可通过延迟注入解决循环依赖。 |
2. 优缺点对比
构造器注入
-
优点:
-
不可变性:依赖字段可声明为
final,确保线程安全和对象状态一致性。 -
明确性:强制要求所有必需依赖,避免
NullPointerException。 -
代码简洁性:结合 Lombok 的
@RequiredArgsConstructor,可自动生成构造方法。 -
兼容测试:易于在单元测试中手动注入依赖。
-
-
缺点:
-
灵活性不足:对可选依赖支持较弱,需通过重载构造方法实现。
-
循环依赖限制:无法处理构造器级别的循环依赖。
-
Setter 注入
-
优点:
-
灵活性高:支持可选依赖,允许动态重新配置依赖。
-
解决循环依赖:Spring 容器可处理 Setter 注入的循环依赖。
-
向后兼容:适合逐步迁移旧代码到依赖注入模式。
-
-
缺点:
-
状态不稳定:对象可能在未完全初始化时被使用(如缺少必需依赖)。
-
线程安全风险:依赖可变性可能导致多线程环境下的问题。
-
3. 适用场景
| 场景 | 推荐方式 | 理由 |
|---|---|---|
| 必需依赖 | 构造器注入 | 强制依赖不可为空,确保对象完全初始化。 |
| 可选依赖 | Setter 注入 | 允许依赖为 null,或通过默认值处理。 |
| 不可变对象 | 构造器注入 | 结合 final 字段,保证线程安全。 |
| 循环依赖 | Setter 注入 | Spring 容器支持通过 Setter 解决循环依赖。 |
| 遗留代码迁移 | Setter 注入 | 逐步引入依赖注入,无需修改现有构造方法。 |
4. 代码示例
构造器注入
java
复制
@Component
public class OrderService {private final PaymentService paymentService; // final 字段@Autowired // Spring 4.3+ 可省略单构造器的 @Autowiredpublic OrderService(PaymentService paymentService) {this.paymentService = paymentService;}
}
Setter 注入
java
复制
@Component
public class UserService {private EmailService emailService;@Autowiredpublic void setEmailService(EmailService emailService) {this.emailService = emailService;}
}
5. 现代实践建议
-
优先使用构造器注入:Spring 官方推荐构造器注入作为默认选择,尤其在 Spring 4.x+ 中,因其对不可变性和安全性的支持。
-
结合 Lombok 简化代码:使用
@RequiredArgsConstructor自动生成构造方法,减少样板代码。 -
谨慎处理循环依赖:循环依赖通常是设计问题的信号,应优先通过重构解决,而非依赖 Setter 注入。
总结
构造器注入和 Setter 注入各有优劣,选择取决于具体需求:
-
构造器注入:适用于强制依赖、不可变对象和高安全性场景。
-
Setter 注入:适用于可选依赖、动态配置或遗留代码迁移。
在实际项目中,可以混合使用两者,但应以构造器注入为主,遵循“依赖明确、状态可靠”的设计原则。
二、最佳实践
在 Spring 的构造器注入中,将依赖字段声明为 final 可以增强线程安全和对象状态一致性,这主要归功于 Java 内存模型(JMM)对 final 字段的特殊处理和 不可变性(Immutability) 的设计原则。以下是具体原因:
1. final 字段的内存可见性保证
根据 Java 内存模型(JSR-133)的规范:
-
初始化安全性:当一个对象被正确构造(即构造方法没有发生
this引用逸出)时,所有线程在访问该对象的final字段时,无需同步即可看到构造方法中初始化的值。 -
禁止指令重排序:JVM 会对
final字段的写操作插入内存屏障,确保构造方法中对final字段的赋值操作不会被重排序到对象引用发布之后。
示例
java
复制
public class OrderService {private final PaymentService paymentService; // final 字段public OrderService(PaymentService paymentService) {this.paymentService = paymentService; // 初始化 final 字段}
}
-
当一个线程创建
OrderService对象后,其他线程在访问paymentService时,一定能看到构造方法中初始化的值,不会出现未初始化或部分初始化的状态。
2. 不可变性(Immutability)
-
字段不可变:
final字段一旦被赋值,其引用不能再被修改(即不能通过setter或其他方法重新赋值)。 -
状态一致性:对象的状态(依赖的组件)在构造完成后即固定,不会因后续代码的意外修改而破坏一致性。
对比 Setter 注入
java
复制
public class UserService {private EmailService emailService; // 非 final 字段public void setEmailService(EmailService emailService) {this.emailService = emailService; // 可能被多次调用或并发修改}
}
-
线程安全问题:如果多线程同时调用
setEmailService,可能导致竞态条件(Race Condition),最终emailService的值可能不一致。 -
状态不一致:对象可能在某个时刻处于“部分初始化”状态(例如,依赖未完全注入)。
3. 避免 this 引用逸出
-
构造器注入的天然优势:在构造方法中完成依赖注入,可以避免在对象未完全初始化前暴露
this引用。 -
final字段的强制约束:必须在构造方法中完成final字段的初始化,否则代码无法编译。这强制开发者保证依赖的完整性。
反例(Setter 注入中的风险)
java
复制
public class UserService {private EmailService emailService;public UserService() {// 构造方法中可能提前暴露 this 引用(错误实践)SomeRegistry.register(this); // 此时 emailService 尚未初始化!}public void setEmailService(EmailService emailService) {this.emailService = emailService;}
}
-
如果其他线程通过
SomeRegistry获取到未完全初始化的UserService实例,可能导致NullPointerException。
4. 实际场景中的线程安全
-
无状态服务:Spring 中的 Bean 默认是单例的,如果 Bean 是无状态的(例如仅依赖其他组件),结合
final字段的不可变性,天然支持多线程并发访问。 -
无需额外同步:由于依赖不可变,无需使用
synchronized或volatile等同步机制。
对比 Setter 注入的线程安全成本
java
复制
public class UserService {private volatile EmailService emailService; // 需要 volatile 保证可见性public synchronized void setEmailService(EmailService emailService) {this.emailService = emailService; // 需要同步锁保证原子性}
}
-
为了线程安全,Setter 注入可能需要额外的同步机制,增加了代码复杂性和性能开销。
总结
通过构造器注入将依赖字段声明为 final,可以从以下层面保证线程安全和状态一致性:
-
内存可见性:JMM 确保
final字段的初始化值对所有线程立即可见。 -
不可变性:依赖引用不可修改,消除竞态条件。
-
初始化完整性:强制依赖在对象创建时完成注入,避免部分初始化状态。
因此,构造器注入 + final 字段 是 Spring 中实现线程安全依赖注入的最佳实践。
相关文章:
Spring 构造器注入和setter注入的比较
一、比较说明 在 Spring 框架中,构造器注入(Constructor Injection)和 Setter 注入(Setter Injection)是实现依赖注入(DI)的两种主要方式。它们的核心区别在于依赖注入的时机、代码设计理念以及…...
如何选择DevOps平台?GitHub、GitLab、BitBucket、Jenkins对比与常见问题解答
本文内容来源github.com,由GitHub中国授权合作伙伴-创实信息进行翻译整理。 欢迎通过021-61210910、customershcsinfo.com联系我们,免费试用GitHub企业版。 软件是当今领先企业的核心,而开发者则是软件的核心。GitHub作为一个完整的开发者平台…...
react中的fiber和初次渲染
源码中定义了不同类型节点的枚举值 组件类型 文本节点HTML标签节点函数组件类组件等等 src/react/packages/react-reconciler/src/ReactWorkTags.js export const FunctionComponent 0; export const ClassComponent 1; export const IndeterminateComponent 2; // Befo…...
闭包+求解候选码+最小函数依赖集
一、闭包 直接上例题 简单明了 A的闭包ABC ABC的闭包ABCD ABCD的闭包ABCDE ABCDE的闭包ABCDEG 等于集合R的全集 所以A的闭包为ABCDEG AB的闭包为ABC 二、候选码 答案: 三、最小函数依赖集 求F的最小函数依赖集 去掉多余的 然后! 化为最简...
DeepSeek 助力 Vue3 开发:打造丝滑的表格(Table)之功能优化,添加表格空状态提示,带插图的空状态,Table7空状态2
前言:哈喽,大家好,今天给大家分享一篇文章!并提供具体代码帮助大家深入理解,彻底掌握!创作不易,如果能帮助到大家或者给大家一些灵感和启发,欢迎收藏关注哦 💕 目录 Deep…...
Unity Shader 学习15:可交互式雪地流程
本质是 利用顶点变换实现的: 通过一个俯视整个场地的正交摄像机,根据绑定在移动物体身上的粒子系统,来获取物体移动过的位置,记录到一张RenderTexture上作为轨迹图,再通过这张图来对雪地做顶点变换。 1. 由于顶点变换需…...
工具介绍《netcat》
nc(netcat)是一款功能强大的网络工具,被称为“网络瑞士军刀”,支持TCP/UDP协议,广泛用于调试、数据传输、端口扫描、网络连接测试等场景。以下是其详细介绍: 一、核心功能 端口扫描 检测目标主机的端口开放…...
嵌入式开发之串行数据处理
前题 前面几篇文章写了关于嵌入式软件开发时,关于串行数据处理的一些相关内容,有兴趣的可以看看《嵌入式开发:软件架构、驱动开发与串行数据处理》、《嵌入式软件开发之生产关系模型》和《嵌入式开发之Modbus-RTU协议解析》相关的内容。从业十…...
Centos的ElasticSearch安装教程
由于我们是用于校园学习,所以最好是关闭防火墙 systemctl stop firewalld systemctl disable firewalld 个人喜欢安装在opt临时目录,大家可以随意 在opt目录下创建一个es-standonely-docker目录 mkdir es-standonely-docker 进入目录编辑yml文件 se…...
SyntaxError: Unexpected token ‘xxx‘
🤍 前端开发工程师、技术日更博主、已过CET6 🍨 阿珊和她的猫_CSDN博客专家、23年度博客之星前端领域TOP1 🕠 牛客高级专题作者、打造专栏《前端面试必备》 、《2024面试高频手撕题》、《前端求职突破计划》 🍚 蓝桥云课签约作者、…...
Unity自定义区域UI滑动事件
自定义区域UI滑动事件 介绍制作1.创建一个Image2.创建脚本 总结 介绍 一提到滑动事件联想到有太多的插件了比如EastTouchBundle,今天想单纯通过UI去做一个滑动事件而不是基于Box2d或者Box去做滑动事件。 制作 1.创建一个Image 2.创建脚本 using UnityEngine; us…...
单链表封装 - 使用JavaScript封装
痛苦就是在蜕变吗 目录 链表:链表的特点:单链表:单链表的封装- JS封装: 单链表的应用:解决回文:解决击鼓传花:十进制进制转换其他进制: 链表: 链表就是一种物理存储单元…...
GET3D:从图像中学习的高质量3D纹理形状的生成模型
【摘要】 本文提出了GET3D,这是一种新的生成模型,能够生成具有任意拓扑结构的高质量3D纹理网格,可以直接被3D渲染引擎使用并在下游应用中立即使用。现有的3D生成模型要么缺乏几何细节,要么生成的网格拓扑受限,通常不支持纹理,或者在生成过程中使用神经渲染器,使得它们在…...
TypeError: Cannot convert object to primitive value
🤍 前端开发工程师、技术日更博主、已过CET6 🍨 阿珊和她的猫_CSDN博客专家、23年度博客之星前端领域TOP1 🕠 牛客高级专题作者、打造专栏《前端面试必备》 、《2024面试高频手撕题》、《前端求职突破计划》 🍚 蓝桥云课签约作者、…...
【uniapp】图片添加canvas水印
目录 需求&背景实现地理位置添加水印 ios补充 需求&背景 需求:拍照后给图片添加水印, 水印包含经纬度、用户信息、公司logo等信息。 效果图: 方案:使用canvas添加水印。 具体实现:上传图片组件是项目里现有的ÿ…...
Flutter——最详细原生交互(MethodChannel、EventChannel、BasicMessageChannel)使用教程
MethodChannel(方法通道) 用途:实现 双向通信,用于调用原生平台提供的 API 并获取返回结果。 场景:适合一次性操作,如调用相机、获取设备信息等。 使用步骤: Flutter 端:通过 Meth…...
如何在PHP爬虫中处理异常情况的详细指南
一、常见的异常类型 在爬虫开发中,可能会遇到以下几种常见的异常情况: 网络请求失败:目标服务器不可用或网络连接问题。 页面结构变化:目标网站更新了HTML结构,导致选择器无法正确匹配。 反爬机制触发:请…...
贪吃蛇身匀速运动模型
通用运动模型 我们已知斜线为移动的距离 d d d, x x x轴总偏移量为 d x dx dx, y y y轴总偏移量为 d y dy dy,在一帧当中,我们也知道能走的距离为 m d md md。那么作为一般的运动模型,该如何确定我们进行移动的方向呢&…...
npm 执行安装报错
Fix the upstream dependency conflict, or retry this command with --force or --legacy-peer-deps to accept an incorrect (and potentially broken) dependency resolution. 原因 主要的原因是 npm7 以上的版本,新增了一个对等依赖的特性,在以…...
SPA单页面应用优化SEO
1.SSR服务端渲染 将组件或页面通过服务器生成html,再返回给浏览器,如nuxt.js或vue-server-renderer const Vue require(vue); const server require(express)(); const renderer require(vue-server-renderer).createRenderer();const vueApp new …...
笔记五:C语言编译链接
Faye:孤独让我们与我们所爱的人相处的每个瞬间都无比珍贵,让我们的回忆价值千金。它还驱使你去寻找那些你在我身边找不到的东西。 ---------《寻找天堂》 目录 一、编译和链接的介绍 1.1 程序的翻译环境和执行环境 1.1.1 翻译环境 1.1.2 运行环境 …...
【c语言概述、数据类型、运算符与表达式精选题】
c语言概述、数据类型、运算符与表达式精选题 一、易错题1.1🎄 c程序的执行1.2🎄 c程序的基本组成单元1.3🎄 c程序的组成1.4🎄 5种基本类型数据类型长度1.5🎄 C语言关键字1.6🎄 整型常量1.7🎄 合…...
200个前卫街头氛围涂鸦艺术水墨颜料手绘笔迹飞溅PNG免扣迭加纹理素材 VANTABLACK TEXTURES
探索 Vantablack 200 纹理包:您获得前卫、高分辨率纹理的首选资源。非常适合旨在为其作品添加原始都市氛围的设计师。这些透明迭加层使用简单,但非常有效,只需单击几下,即可将您的设计从普通变为非凡。准备好用既酷又百搭的质地来…...
机试准备第11天
第一题是浮点数加法,目前写过最长的代码。 #include <stdio.h> #include <string> #include <iostream> #include <vector> using namespace std; int main() {string str1;string str2;while (getline(cin, str1) && getline(cin…...
OpenIndiana Hipster系统安装配置
gcc安装 直接pkt install gcc会报错 需要 先pkt update,然后重启(不重启还是报错)用pkg search compiler找到可用的gcc包再pkt install xx安装这个包 TCP配置 参考这个网站:https://community.spiceworks.com/t/setting-tcp-p…...
深度学习模型Transformer核心组件—自注意力机制
第一章:人工智能之不同数据类型及其特点梳理 第二章:自然语言处理(NLP):文本向量化从文字到数字的原理 第三章:循环神经网络RNN:理解 RNN的工作机制与应用场景(附代码) 第四章:循环神经网络RNN、LSTM以及GR…...
Java核心语法:从变量到控制流
一、变量与数据类型(对比Python/C特性) 1. 变量声明三要素 // Java(强类型语言,需显式声明类型) int age 25; String name "CSDN"; // Python(动态类型) age 25 name …...
nature genetics | SCENT:单细胞多模态数据揭示组织特异性增强子基因图谱,并可识别致病等位基因
–https://doi.org/10.1038/s41588-024-01682-1 Tissue-specific enhancer–gene maps from multimodal single-cell data identify causal disease alleles 研究团队和单位 Alkes L. Price–Broad Institute of MIT and Harvard Soumya Raychaudhuri–Harvard Medical S…...
大白话如何使用 CSS 实现响应式布局?请列举一些常见的方法。
大白话如何使用 CSS 实现响应式布局?请列举一些常见的方法。 答题思路 首先要解释什么是响应式布局,让读者明白其概念和重要性。然后依次介绍常见的实现响应式布局的CSS方法,包括媒体查询、弹性布局(Flexbox)、网格布…...
基于数据挖掘的疾病数据可视化分析与预测系统
【大数据】基于数据挖掘的疾病数据可视化分析与预测系统(完整系统源码开发笔记详细部署教程)✅ 目录 一、项目简介二、项目界面展示三、项目视频展示 一、项目简介 📌 技术核爆点:✔️ Python全栈开发Flask高能框架 ✔️ 爬虫技术…...
