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

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,可以从以下层面保证线程安全和状态一致性:

  1. 内存可见性:JMM 确保 final 字段的初始化值对所有线程立即可见。

  2. 不可变性:依赖引用不可修改,消除竞态条件。

  3. 初始化完整性:强制依赖在对象创建时完成注入,避免部分初始化状态。

因此,构造器注入 + 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添加水印。 具体实现:上传图片组件是项目里现有的&#xff…...

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天

第一题是浮点数加法&#xff0c;目前写过最长的代码。 #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&#xff0c;然后重启&#xff08;不重启还是报错&#xff09;用pkg search compiler找到可用的gcc包再pkt install xx安装这个包 TCP配置 参考这个网站&#xff1a;https://community.spiceworks.com/t/setting-tcp-p…...

深度学习模型Transformer核心组件—自注意力机制

第一章&#xff1a;人工智能之不同数据类型及其特点梳理 第二章&#xff1a;自然语言处理(NLP)&#xff1a;文本向量化从文字到数字的原理 第三章&#xff1a;循环神经网络RNN&#xff1a;理解 RNN的工作机制与应用场景(附代码) 第四章&#xff1a;循环神经网络RNN、LSTM以及GR…...

Java核心语法:从变量到控制流

一、变量与数据类型&#xff08;对比Python/C特性&#xff09; 1. 变量声明三要素 // Java&#xff08;强类型语言&#xff0c;需显式声明类型&#xff09; int age 25; String name "CSDN"; // Python&#xff08;动态类型&#xff09; 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 实现响应式布局&#xff1f;请列举一些常见的方法。 答题思路 首先要解释什么是响应式布局&#xff0c;让读者明白其概念和重要性。然后依次介绍常见的实现响应式布局的CSS方法&#xff0c;包括媒体查询、弹性布局&#xff08;Flexbox&#xff09;、网格布…...

基于数据挖掘的疾病数据可视化分析与预测系统

【大数据】基于数据挖掘的疾病数据可视化分析与预测系统&#xff08;完整系统源码开发笔记详细部署教程&#xff09;✅ 目录 一、项目简介二、项目界面展示三、项目视频展示 一、项目简介 &#x1f4cc; 技术核爆点&#xff1a;✔️ Python全栈开发Flask高能框架 ✔️ 爬虫技术…...