Spring 依赖 详解
Spring 依赖详解
在 Spring 框架中,依赖 是指一个对象(Bean)需要另一个对象(Bean)来完成其功能的情况。Spring 通过 依赖注入(Dependency Injection, DI) 和 控制反转(Inversion of Control, IoC) 来实现对依赖的管理。
1. 什么是依赖?
1.1 概念
- 依赖 是指一个类需要另一个类的协助才能完成其工作。
- 例如,
OrderService
可能依赖于OrderRepository
来访问数据库。
1.2 传统方式的问题
- 在传统开发中,依赖通过手动创建实例(
new
)来实现。 - 问题:
- 强耦合:代码依赖具体实现,难以扩展和替换。
- 难以测试:无法轻松替换依赖的 Mock 对象。
- 复杂管理:在大型项目中,依赖关系复杂且难以维护。
2. Spring 依赖的管理
Spring 使用 IoC 容器来管理依赖,通过 依赖注入 将依赖关系注入到对象中,解决传统方式的问题。
2.1 依赖注入(DI)的核心思想
- 对象本身不负责管理其依赖的创建,而是由外部容器注入。
- 依赖关系在配置文件(XML、JavaConfig)或注解中声明。
3. 依赖注入的方式
Spring 提供三种主要的依赖注入方式:
3.1 构造器注入
- 概念:通过构造方法注入依赖。
- 优点:
- 强制依赖注入,避免遗漏。
- 有助于实现不可变对象。
- 示例:
@Component public class OrderService {private final OrderRepository orderRepository;@Autowiredpublic OrderService(OrderRepository orderRepository) {this.orderRepository = orderRepository;} }
3.2 Setter 方法注入
- 概念:通过 Setter 方法注入依赖。
- 优点:
- 灵活性高,可以在运行时动态替换依赖。
- 适用于可选依赖。
- 示例:
@Component public class OrderService {private OrderRepository orderRepository;@Autowiredpublic void setOrderRepository(OrderRepository orderRepository) {this.orderRepository = orderRepository;} }
3.3 字段注入
- 概念:直接在字段上注入依赖。
- 优点:
- 代码简洁,省略了 Getter 和 Setter。
- 缺点:
- 不支持依赖的不可变性。
- 不便于单元测试。
- 示例:
@Component public class OrderService {@Autowiredprivate OrderRepository orderRepository; }
4. 依赖的配置方式
Spring 提供以下几种方式配置依赖关系:
4.1 基于注解的配置(推荐)
- 使用注解标记类和依赖关系。
- 主要注解:
@Component
:声明一个类是 Spring 容器中的 Bean。@Autowired
:自动注入依赖。@Qualifier
:指定具体的 Bean。@Primary
:优先注入特定 Bean。
- 示例:
@Component public class OrderRepository {}@Component public class OrderService {@Autowiredprivate OrderRepository orderRepository; }
4.2 基于 XML 的配置
- 在 XML 文件中定义 Bean 和依赖关系。
- 示例:
<beans><bean id="orderRepository" class="com.example.OrderRepository"/><bean id="orderService" class="com.example.OrderService"><property name="orderRepository" ref="orderRepository"/></bean> </beans>
4.3 基于 JavaConfig 的配置
- 使用 Java 类和注解定义 Bean 和依赖关系。
- 示例:
@Configuration public class AppConfig {@Beanpublic OrderRepository orderRepository() {return new OrderRepository();}@Beanpublic OrderService orderService(OrderRepository orderRepository) {return new OrderService(orderRepository);} }
5. 自动装配(Autowired)
5.1 @Autowired
的工作原理
- Spring 通过类型匹配自动注入依赖。
- 可以与
@Qualifier
或@Primary
配合使用,以解决多个候选 Bean 的问题。
5.2 示例
@Component
public class OrderService {@Autowiredprivate OrderRepository orderRepository;
}
6. 多 Bean 配置与冲突解决
当存在多个类型相同的 Bean 时,Spring 提供以下解决方案:
6.1 使用 @Qualifier
- 明确指定注入的 Bean。
- 示例:
@Component("repo1") public class OrderRepository {}@Component("repo2") public class BackupRepository {}@Component public class OrderService {@Autowired@Qualifier("repo1")private OrderRepository orderRepository; }
6.2 使用 @Primary
- 设置默认的优先级 Bean。
- 示例:
@Component @Primary public class OrderRepository {}
7. 作用域(Scope)
Spring 中 Bean 的默认作用域是 单例(Singleton),还支持其他作用域:
- Singleton:整个应用中仅有一个实例(默认)。
- Prototype:每次获取时创建新的实例。
- Request:每个 HTTP 请求一个实例(Web 应用)。
- Session:每个 HTTP 会话一个实例(Web 应用)。
- Application:每个 ServletContext 一个实例(Web 应用)。
示例
@Component
@Scope("prototype")
public class OrderService {}
8. 循环依赖
8.1 什么是循环依赖?
- 两个或多个 Bean 互相依赖,形成循环。
- 示例:
@Component public class A {@Autowiredprivate B b; }@Component public class B {@Autowiredprivate A a; }
8.2 Spring 的解决方式
- 单例模式:
- Spring 使用三级缓存解决循环依赖:
- 一级缓存:完整实例(单例池)。
- 二级缓存:半成品实例。
- 三级缓存:对象工厂,用于创建代理对象。
- Spring 使用三级缓存解决循环依赖:
- 原型模式:
- 不支持循环依赖,抛出异常。
8.3 避免循环依赖的建议
- 重构代码,避免互相注入。
- 使用
@Lazy
注解延迟加载。 - 使用构造器注入时确保无循环依赖。
9. 依赖的生命周期
Spring Bean 的生命周期由容器管理,主要包括以下阶段:
- 实例化:通过反射创建对象。
- 依赖注入:注入依赖对象。
- 初始化:执行初始化方法。
- 使用:Bean 被调用。
- 销毁:容器关闭时调用销毁方法。
示例
@Component
public class OrderService {@PostConstructpublic void init() {System.out.println("Initializing OrderService");}@PreDestroypublic void destroy() {System.out.println("Destroying OrderService");}
}
10. Spring 依赖的优缺点
10.1 优点
- 降低耦合:Bean 的创建和管理由容器负责,模块更加独立。
- 便于测试:可以轻松替换依赖为 Mock 对象。
- 增强灵活性:可以动态配置和替换 Bean。
- 代码简洁:减少手动管理依赖的代码。
10.2 缺点
- 学习曲线:需要理解 IoC 和 DI 的概念及实现。
- 配置复杂性:在大型项目中,Bean 和依赖关系可能变得复杂。
- 性能开销:动态代理和反射可能会影响性能。
11. 总结
Spring 的依赖管理通过 IoC 容器和依赖注入,大幅提升了代码的灵活性、可读性和可维护性。通过多种注入方式(构造器、Setter、字段),以及配置方式(注解、XML、JavaConfig),Spring 适应了
各种开发场景。深入理解 Spring 依赖的机制和原理,有助于开发者设计更优雅、解耦的系统架构。
相关文章:
Spring 依赖 详解
Spring 依赖详解 在 Spring 框架中,依赖 是指一个对象(Bean)需要另一个对象(Bean)来完成其功能的情况。Spring 通过 依赖注入(Dependency Injection, DI) 和 控制反转(Inversion of…...
千益畅行,旅游卡有些什么优势?
千益畅行共享旅游卡是一种创新的旅游服务模式,旨在通过整合各类旅游资源,为用户提供一站式的旅游解决方案。这张旅游卡支持2至6人同行,涵盖了接机、酒店、用餐、大巴、导游、景区门票等服务,用户只需自行承担往返交通费用即可享受…...
Ubuntu24 cgroupv2导致rancher(k3s)启动失败的处理
方案一: 修改系统镜像为ubuntu18 方案二: 修改当前系统的cgroup版本,由v2改成v1 修改步骤: 1、查看当前cgroup版本 stat -fc %T /sys/fs/cgroup cgroup v2,输出结果为cgroup2fs cgroup v1,输出为tm…...
学习CSS第二天
学习文章目录 一.内部样式 一.内部样式 写在 html 页面内部,将所有的 CSS 代码提取出来,单独放在 <style> 标签中 语法: <style> h1 { color: red; font-size: 40px; } </style>注意点: <style> 标签理…...

2021数学分析【南昌大学】
2021 数学分析 求极限 lim n → ∞ 1 n ( n + 1 ) ( n + 2 ) ⋯ ( n + n ) n \lim_{n \to \infty} \frac{1}{n} \sqrt [n]{(n+1)(n+2) \cdots (n+n)} n→∞limn1n(n+1)(n+2)⋯(n+n) lim n → ∞ 1 n ( n + 1 ) ( n + 2 ) ⋯ ( n + n ) n = lim n → ∞ ( n + …...

单端和差分信号的接线法
内容来源:【单端信号 差分信号与数据采集卡的【RSE】【 NRES】【 DIFF】 模式的连接】 此篇文章仅作笔记分享。 单端输入 单端信号指的是输入信号由一个参考端和一个信号端构成,参考端一般是地端,信号就是通过计算信号端口和地端的差值所得…...

力扣-图论-2【算法学习day.52】
前言 ###我做这类文章一个重要的目的还是给正在学习的大家提供方向和记录学习过程(例如想要掌握基础用法,该刷哪些题?)我的解析也不会做的非常详细,只会提供思路和一些关键点,力扣上的大佬们的题解质量是非…...
MySQL如何区分幻读和不可重复读
在MySQL中,幻读和不可重复读都是并发事务中可能出现的问题,但它们的表现和原因略有不同。 不可重复读 (Non-Repeatable Read) 不可重复读是指在同一个事务内,多次读取同一行数据时,可能会得到不同的结果。这种情况发生在一个事务…...

界面控件Syncfusion Essential Studio®现在已完全支持 .NET 9
Syncfusion Essential Studio现在完全支持 .NET 9,可最新版本2024 Volume 3 版本中使用!通过此更新,Blazor、.NET MAUI、WPF、WinForms、WinUI和ASP.NET Core 平台中的 Syncfusion 组件以及文档处理库已准备好让您利用 .NET 9 中的最新功能。…...

openEuler安装lsb_release
lsb_release是linux下查看发行版信息用的工具 lsb_release只是一个小程序,它的包名并不是lsb_release lsb_release其实是红帽的一个项目,其名为redhat-lsb 我们的lsb_release就是其中的一部分,更准确的说是redhat-lsb-core 从网站࿱…...

统计数字字符个数
统计数字字符个数 C语言实现C实现Java实现Python实现 💐The Begin💐点点关注,收藏不迷路💐 输入一行字符,统计出其中数字字符的个数。 输入 一行字符串,总长度不超过255。 输出 出为1行,输出…...

44页PDF | 信息化战略规划标准框架方法论与实施方法(限免下载)
一、前言 这份报告详细介绍了企业信息化战略规划的标准框架、方法论以及实施方法,强调了信息化规划应以业务战略和IT战略为驱动力,通过构筑企业架构(EA)来连接长期战略和信息化建设。报告提出了信息化规划原则,探讨了…...

计算机网络期末复习-part1-概述
1、互联网的组成 互联网由两大块组成。 1、边沿部分:由所有连接在互联网上的主机组成,是用户直接使用的部分。 2、核心部分,由大量网络和路由器组成,为边缘部分提供服务。 2、数据传送阶段的三种交换方式的主要特点 1、电路交…...

A1228 php+Mysql旅游供需平台的设计与实现 导游接单 旅游订单 旅游分享网站 thinkphp框架 源码 配置 文档 全套资料
旅游供需平台 1.项目描述2. 开发背景与意义3.项目功能4.界面展示5.源码获取 1.项目描述 随着社会经济的快速发展,生活水平的提高,人们对旅游的需求日益增强,因此,为给用户提供一个便利的查看导游信息,进行导游招募的平…...
RabbitMQ消息可靠性保证机制5--消息幂等性处理
RabbitMQ层面有实现“去重机制”来保证“恰好一次”吗?答案是没并没有,而且现在主流的消息中间件都没有实现。 一般解决重复消息的办法是:在消费端让我们消费消息操作具有幂等性。 幂等性问题并不是消息系统独有,而是࿰…...

Claude3.5如何使用?
Claude 3.5 Sonnet,性能直接吊打了 GPT-4o,甚至价格还更便宜。网友们纷纷展开实测,有人表示自己一半的工作已经可以由它替代了!而最让人惊喜的新功能,莫过于 Artifacts 了。 就在昨天,Anthropic 深夜发布了…...
力扣刷题TOP101:14.BM16 删除有序链表中重复的元素-II
目录: 目的 思路 复杂度 记忆秘诀 python代码 目的 1→1→2→3→3 删除重复后变成2。 思路 这个任务是删除链表里重复的节点包含本身。可以看成是一个抽奖活动的系统升级。某人通过多种方式报名(节点不同),后台数据检测到这些…...

解决github网络慢的问题
前言 本文采用替换host的方式来加速github的git请求,主要我自己用来备份的懒人方式,不然每次都要手动修改hosts文件,skrskrskr… 一、获取到可用的ip 先到这个网站查询到低延迟的ip 站长工具:https://ping.chinaz.com/ 第2步&…...
docker及docker exec命令学习笔记
docker exec 是一个常用的 Docker 命令,允许你在已经运行的容器中执行命令或启动新的进程。以下是详细介绍和常见用法: 基本语法 docker exec [OPTIONS] CONTAINER COMMAND [ARG...]参数详解 1. CONTAINER指定目标容器的名字或容器 ID。可以通过以下命…...

linux环境宝塔服务部署安装及介绍
一、简介 宝塔面板是一款服务器管理软件,支持windows和linux系统,可以通过Web端轻松管理服务器,提升运维效率。例如:创建管理网站、FTP、数据库,拥有可视化文件管理器,可视化软件管理器,可视化C…...
CVPR 2025 MIMO: 支持视觉指代和像素grounding 的医学视觉语言模型
CVPR 2025 | MIMO:支持视觉指代和像素对齐的医学视觉语言模型 论文信息 标题:MIMO: A medical vision language model with visual referring multimodal input and pixel grounding multimodal output作者:Yanyuan Chen, Dexuan Xu, Yu Hu…...

简易版抽奖活动的设计技术方案
1.前言 本技术方案旨在设计一套完整且可靠的抽奖活动逻辑,确保抽奖活动能够公平、公正、公开地进行,同时满足高并发访问、数据安全存储与高效处理等需求,为用户提供流畅的抽奖体验,助力业务顺利开展。本方案将涵盖抽奖活动的整体架构设计、核心流程逻辑、关键功能实现以及…...
反向工程与模型迁移:打造未来商品详情API的可持续创新体系
在电商行业蓬勃发展的当下,商品详情API作为连接电商平台与开发者、商家及用户的关键纽带,其重要性日益凸显。传统商品详情API主要聚焦于商品基本信息(如名称、价格、库存等)的获取与展示,已难以满足市场对个性化、智能…...
镜像里切换为普通用户
如果你登录远程虚拟机默认就是 root 用户,但你不希望用 root 权限运行 ns-3(这是对的,ns3 工具会拒绝 root),你可以按以下方法创建一个 非 root 用户账号 并切换到它运行 ns-3。 一次性解决方案:创建非 roo…...

基于Docker Compose部署Java微服务项目
一. 创建根项目 根项目(父项目)主要用于依赖管理 一些需要注意的点: 打包方式需要为 pom<modules>里需要注册子模块不要引入maven的打包插件,否则打包时会出问题 <?xml version"1.0" encoding"UTF-8…...
Rust 异步编程
Rust 异步编程 引言 Rust 是一种系统编程语言,以其高性能、安全性以及零成本抽象而著称。在多核处理器成为主流的今天,异步编程成为了一种提高应用性能、优化资源利用的有效手段。本文将深入探讨 Rust 异步编程的核心概念、常用库以及最佳实践。 异步编程基础 什么是异步…...
【碎碎念】宝可梦 Mesh GO : 基于MESH网络的口袋妖怪 宝可梦GO游戏自组网系统
目录 游戏说明《宝可梦 Mesh GO》 —— 局域宝可梦探索Pokmon GO 类游戏核心理念应用场景Mesh 特性 宝可梦玩法融合设计游戏构想要素1. 地图探索(基于物理空间 广播范围)2. 野生宝可梦生成与广播3. 对战系统4. 道具与通信5. 延伸玩法 安全性设计 技术选…...

Mac下Android Studio扫描根目录卡死问题记录
环境信息 操作系统: macOS 15.5 (Apple M2芯片)Android Studio版本: Meerkat Feature Drop | 2024.3.2 Patch 1 (Build #AI-243.26053.27.2432.13536105, 2025年5月22日构建) 问题现象 在项目开发过程中,提示一个依赖外部头文件的cpp源文件需要同步,点…...

AirSim/Cosys-AirSim 游戏开发(四)外部固定位置监控相机
这个博客介绍了如何通过 settings.json 文件添加一个无人机外的 固定位置监控相机,因为在使用过程中发现 Airsim 对外部监控相机的描述模糊,而 Cosys-Airsim 在官方文档中没有提供外部监控相机设置,最后在源码示例中找到了,所以感…...

C/C++ 中附加包含目录、附加库目录与附加依赖项详解
在 C/C 编程的编译和链接过程中,附加包含目录、附加库目录和附加依赖项是三个至关重要的设置,它们相互配合,确保程序能够正确引用外部资源并顺利构建。虽然在学习过程中,这些概念容易让人混淆,但深入理解它们的作用和联…...