手把手教你用Java实现IoC,轻松搞定依赖注入!
想要掌握Java的IoC(控制反转)和依赖注入(DI)?本文手把手教你搞定!IoC将对象创建和依赖管理交给外部容器,DI则通过构造器、Setter或接口注入实现。以Spring为例,我们讲解如何配置容器、定义Bean,并使用@Autowired进行依赖注入,包括构造器注入(推荐)、Setter注入(可选)和字段注入(不推荐)。同时,还会介绍Java配置方式,使用@Configuration和@Bean定义Bean。本文还将深入探讨Spring IoC容器的启动流程,从资源定位、加载解析到Bean的实例化、注入和初始化,并详细解析循环依赖问题及其解决方案——三级缓存机制。最后,对比构造器、Setter和字段三种注入方式的优缺点,助你选择最合适的依赖注入策略,提升代码质量。
IoC的核心是将对象创建和依赖管理交给外部容器,DI通过构造器、Setter或接口注入实现。Spring实现DI需配置容器并定义Bean,使用@Autowired进行注入,可通过构造器(推荐)、Setter(可选)或字段(不推荐)完成。Java配置用@Configuration和@Bean定义Bean。启动流程包括定位资源、加载解析为BeanDefinition、注册、实例化、注入、初始化至就绪状态。循环依赖通过三级缓存解决:一级存完整Bean,二级存早期Bean,三级存ObjectFactory,仅支持单例Bean。构造器注入保证必需依赖,Setter用于可选依赖,字段注入破坏封装应避免。

IoC(控制反转)的核心在于将对象的创建和依赖关系的管理权从对象自身转移到外部容器。依赖注入(DI)则是实现IoC的一种常用方式,它通过构造器注入、Setter方法注入或接口注入,将依赖对象“注入”到目标对象中,而非由目标对象主动创建。简单来说,就是让容器帮你new对象,并把对象需要的“零件”也给你装好。

解决方案

Java中实现IoC/DI,通常会借助第三方框架,例如Spring、Guice、Dagger等。这里以Spring为例,讲解如何实现依赖注入:

使用Spring容器: 首先,你需要配置Spring容器。这通常通过XML配置、注解配置或者Java配置来实现。
定义Bean: 你需要定义哪些类由Spring容器来管理,这些类被称为Bean。可以使用
@Component、@Service、@Repository、@Controller等注解来标记Bean,或者在XML配置中声明Bean。依赖注入: Spring提供了三种主要的依赖注入方式:
构造器注入: 通过类的构造器来注入依赖。使用
@Autowired注解在构造器上,Spring会自动找到匹配的Bean并注入。@Component public class MyService { private final MyRepository myRepository; @Autowired public MyService(MyRepository myRepository) { this.myRepository = myRepository; } // ... }Setter方法注入: 通过Setter方法来注入依赖。使用
@Autowired注解在Setter方法上。@Component public class MyService { private MyRepository myRepository; @Autowired public void setMyRepository(MyRepository myRepository) { this.myRepository = myRepository; } // ... }字段注入: 直接在字段上使用
@Autowired注解。虽然方便,但不太推荐,因为它破坏了类的封装性,并且在单元测试时可能会遇到问题。@Component public class MyService { @Autowired private MyRepository myRepository; // ... }
使用
@Qualifier解决歧义: 如果存在多个相同类型的Bean,Spring不知道应该注入哪个,这时可以使用@Qualifier注解来指定具体的Bean名称。```java @Component public class MyService { private final MyRepository myRepository; @Autowired public MyService(@Qualifier("myRepositoryImpl1") MyRepository myRepository) { this.myRepository = myRepository; } // ... } ```Java配置: 除了XML和注解,还可以使用Java配置来定义Bean和依赖关系。使用
@Configuration注解标记配置类,使用@Bean注解标记Bean的创建方法。```java @Configuration public class AppConfig { @Bean public MyRepository myRepository() { return new MyRepositoryImpl(); } @Bean public MyService myService(MyRepository myRepository) { return new MyService(myRepository); } } ```
Spring IoC容器启动流程是怎样的?
Spring IoC容器的启动流程可以大致分为以下几个步骤:
- 定位资源: 容器首先需要定位到Bean定义的资源,例如XML配置文件、注解标记的类等。
- 加载资源: 容器加载这些资源,并将Bean定义解析成Spring内部的BeanDefinition对象。
- 注册BeanDefinition: 容器将BeanDefinition注册到BeanDefinitionRegistry中,这是一个Bean定义的注册中心。
- 实例化Bean: 容器根据BeanDefinition创建Bean实例。这通常发生在第一次请求Bean时,也可以配置成在容器启动时就预先实例化。
- 依赖注入: 容器将Bean所需的依赖注入到Bean实例中。
- 初始化Bean: 容器调用Bean的初始化方法(如果配置了),例如实现了
InitializingBean接口的afterPropertiesSet()方法,或者使用@PostConstruct注解标记的方法。 - Bean准备就绪: Bean实例创建完成,可以被应用程序使用了。
构造器注入、Setter注入和字段注入,应该选择哪种?
这三种注入方式各有优缺点:
- 构造器注入: 强制依赖,保证对象创建时所有必需的依赖都已就绪。有利于对象的不可变性。推荐使用。
- Setter注入: 允许可选依赖,可以在对象创建后动态设置依赖。但容易出现依赖未设置的情况。
- 字段注入: 最简单,但破坏了封装性,不利于单元测试。不推荐使用。
总体来说,构造器注入是最佳选择,因为它能够保证对象的完整性和不可变性。如果存在可选依赖,可以考虑Setter注入。尽量避免字段注入。
循环依赖问题如何解决?
循环依赖是指两个或多个Bean之间相互依赖,形成一个循环引用。例如,A依赖B,B依赖C,C又依赖A。Spring IoC容器可以通过以下方式解决循环依赖问题:
- 提前暴露Bean: Spring在创建Bean时,会提前将Bean的ObjectFactory暴露出来,以便其他Bean可以引用。这样,即使Bean还没有完全创建完成,也可以被其他Bean引用。
- 三级缓存: Spring使用三级缓存来解决循环依赖问题。
- 一级缓存 (singletonObjects): 存放完全初始化好的Bean。
- 二级缓存 (earlySingletonObjects): 存放提前暴露的Bean,这些Bean可能还没有完全初始化完成。
- 三级缓存 (singletonFactories): 存放Bean的ObjectFactory,用于创建Bean实例。
当出现循环依赖时,Spring会首先尝试从一级缓存中获取Bean,如果获取不到,则尝试从二级缓存中获取,如果还获取不到,则从三级缓存中获取ObjectFactory,并使用ObjectFactory创建Bean实例。创建Bean实例后,将Bean放入二级缓存,并移除三级缓存中的ObjectFactory。当Bean完全初始化完成后,将Bean从二级缓存移动到一级缓存。
需要注意的是,Spring只能解决单例Bean的循环依赖问题,对于原型Bean,由于每次获取Bean都是一个新的实例,因此无法解决循环依赖问题。
以上就是本文的全部内容了,是否有顺利帮助你解决问题?若是能给你带来学习上的帮助,请大家多多支持golang学习网!更多关于文章的相关知识,也可关注golang学习网公众号。
桌面快捷方式没了?简单几步教你快速找回!
- 上一篇
- 桌面快捷方式没了?简单几步教你快速找回!
- 下一篇
- PHP小白都能懂的Memcached实战教程,超简单!
-
- 文章 · java教程 | 15小时前 | map · 并发安全 · 缓存设计 · Java教程 · java optional concurrenthashmap computeIfAbsent Map缓存
- Java computeIfAbsent 缓存初始化实战:少写判断、避开空值和并发坑
- 236浏览 收藏
-
- 文章 · java教程 | 1天前 | Java · 异步编程 · 后端开发 · CompletableFuture · 接口聚合 · java 结果合并 completablefuture 并行调用 超时兜底
- Java CompletableFuture 多接口聚合完整流程:并行调用、超时兜底和结果合并
- 428浏览 收藏
-
- 文章 · java教程 | 1天前 | Java · 线程安全 · DateTimeFormatter · 日期处理 · 并发问题 · java 线程安全 日期格式化 threadlocal SimpleDateFormat DateTimeFormatter
- Java SimpleDateFormat 日期偶发错乱怎么办:从共享实例到线程安全一步步排查
- 481浏览 收藏
-
- 文章 · java教程 | 3天前 | http接口 · httpclient · Java教程 · 接口调试 · 超时处理 · java 接口调用 httpclient 超时控制 状态码 响应体
- Java HttpClient 调接口实战:超时、状态码和响应体这样处理
- 224浏览 收藏
-
- 文章 · java教程 | 3天前 | 时间处理 · instant · Java教程 · 时区转换 · DateTimeFormatter · java DateTimeFormatter java.time 时区处理 ZoneId INSTANT
- Java 时间与时区处理实战:Instant、ZoneId 和 DateTimeFormatter 怎么配
- 461浏览 收藏
-
- 文章 · java教程 | 3天前 | Java · Stream · 集合统计 · 分组聚合 · Collectors · java Stream Collectors groupingBy counting summarizingInt
- Java Stream 分组统计实战:groupingBy、counting 和 summarizingInt 怎么用
- 478浏览 收藏
-
- 文章 · java教程 | 3天前 | Java · 文件读取 · 异常处理 · 资源管理 · try-with-resources · java 异常处理 try-with-resources 资源关闭 AutoCloseable 文件流
- Java try-with-resources 资源关闭实战:文件流和目录扫描这样写更稳
- 268浏览 收藏
-
- 文章 · java教程 | 4天前 | Java教程 · 后端开发 · BigDecimal · 金额计算 · java 舍入 bigdecimal 浮点误差 金额计算 RoundingMode
- Java BigDecimal 金额计算实战:避免浮点误差和舍入问题
- 324浏览 收藏
-
- 文章 · java教程 | 4天前 | 异步编程 · Java教程 · 超时治理 · CompletableFuture · java 异步任务 超时处理 completablefuture orTimeout completeOnTimeout
- Java CompletableFuture 超时处理实战:orTimeout 和兜底结果怎么选
- 421浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 485次学习
-
- ljg-skills
- ljg-skills 是李继刚开源的 AI 技能与提示词集合,面向大模型使用者整理了一批可复用的 prompt、角色设定和任务技能模板,适合用于学习提示词设计、搭建个人 AI 工作流和沉淀团队常用智能体能力。
- 184次使用
-
- MELO音乐
- MELO音乐是一站式AI视频与音乐制作助手,对标suno, udio的高品质体验。提供伴奏生成、原创写词、无损导出、哼唱识曲、混音变声等全套音频与短视频编辑工具。无论是流行Kpop、电音说唱、民谣古风、摇滚儿歌还是商用轻音乐,MELO为你免费谱曲,轻松做同款!
- 200次使用
-
- UniScribe
- UniScribe 是一款 AI 音视频转文字与内容整理工具,支持上传音频、视频文件或粘贴 YouTube 链接,自动生成转写文本、摘要、思维导图和关键问题,并支持多格式导出,适合会议记录、课程学习、访谈整理和内容创作复盘。
- 178次使用
-
- 剧云
- 剧云是专业中文剧本创作平台,安全稳定运行十余年,集成AI编剧、剧本医生审核、人物小传、剧情关系图、大纲编写、多人协作、Word导入导出、版权管控功能,数据安全防护,轻松高效创作剧本。
- 341次使用
-
- 万象有声
- 万象有声,一个专为有声创作者打造的新一代智能有声内容创作平台。平台提供专业的智能拆章、智能画本编辑、AI配音、AI生成音效、后期制作、智能对轨、智能审听等有声创作全流程工具,可以帮助创作者高效、低成本创作出引人入胜的有声作品。立即体验,让有声书制作更简单!
- 339次使用
-
- 提升Java功能开发效率的有力工具:微服务架构
- 2023-10-06 501浏览
-
- 掌握Java海康SDK二次开发的必备技巧
- 2023-10-01 501浏览
-
- 如何使用java实现桶排序算法
- 2023-10-03 501浏览
-
- Java开发实战经验:如何优化开发逻辑
- 2023-10-31 501浏览
-
- 如何使用Java中的Math.max()方法比较两个数的大小?
- 2023-11-18 501浏览

