Spring依赖注入与反转控制到底是个啥?
目录
1. 引言
2. 管中窥豹
3.1 Spring 依赖注入
3.2 Bean 的依赖注入方式有两种
4. 总结
1. 引言
此文目的是用通俗易懂的语言讲清楚什么是依赖注入与反转控制,在看了大量的博客文章后归纳总结,便于后续巩固!我相信,大多数人像我一样,刚开始学习Spring框架的时候仅仅是在学习技术,记住如何做而非为何如此做。因此,本文结合生活例子与项目实际问题入手,讲清楚Spring入门的两个概念。其中,第二篇章管中窥豹需要具备Spring基础知识,你不用非常理解但至少你知道怎么写依赖注入。
2. 管中窥豹
话说,有一天蟹老板想开个餐馆,可是开什么店好呢?他列出了几个选择,牛排店、蟹肉煲店、炸鸡店,火锅店,他非常纠结。于是海绵宝宝告诉他你可以春天卖牛排、夏天卖蟹肉堡、秋天换卖炸鸡,冬天卖火锅。于是,蟹老板接纳了海绵宝宝的意见。春天到了,蟹老板叫海绵宝宝开发了一套店内点餐系统,于是海绵宝宝定义了如下代码:
public interface Chef {public void cook(); // 厨师长做菜public void sing(); // 厨师长唱歌
}
并且从海底世界招聘到了张三厨师长,张三厨师长很擅长做牛排,尤其是黑椒牛排一绝,他也喜欢唱歌,因此海绵宝宝写下了如下代码:
public class ZhangChef implements Chef {String name;int age;public ZhangChef(String name, int age) {this.name = name;this.age = age;}@Overridepublic void cook() {System.out.println(name+"正在做黑椒牛排...");}@Overridepublic void sing() {System.out.println(name+"唱着伤不起...");}
}
最后,在我们的蟹老板的英明带领下,嘉年华顺利开店:
public class Restaurant {private String name;private String specialty;Restaurant(String name, String specialty) {this.name = name;this.specialty = specialty;}public void kitchen(){ZhangChef zhang = new ZhangChef("张三",26);System.out.println("我们的厨师长是:"+zhang.name+",今年高龄:"+zhang.age);zhang.cook(); //厨师做菜}}
运行结果>>> 我们的厨师长是:张三,今年高龄:26张三正在做黑椒牛排...
这三个月过的很快,蟹老板也赚了很多钱,海绵宝宝更是成为了本店的CTO,大家其乐融融,一片安好。可是明天就到了夏天,餐厅更换业务,开始经营蟹肉煲,于是张三厨师长被毕业了,李四厨师长正好擅长蟹肉煲,我们的厨师长现在换成了李四。晚上十点钟,海绵宝宝正在加班加点改代码,他需要把整个系统的厨师长换成李四,这是他换好的第一个文件:
public class LiChef implements Chef {String name;int age;public LiChef(String name, int age) {this.name = name;this.age = age;}@Overridepublic void cook() {System.out.println(name+"正在做蟹肉煲...");}@Overridepublic void sing() {System.out.println(name+"唱着撒浪嘿呦...");}
}
然后,他需要把整个系统所有方法的zhangChef换成LisiChef,海绵宝宝工作了一夜,第二天依旧没有改完,直到改了半个月后才把整个系统的Chef换完,于是海绵宝宝丢了CTO的职位,成为了普通打工仔程序员。海绵宝宝很不甘心,他辞去了工作,专心研究这个问题如何解决。他发现,Chef 类需要在 Restaurant 类中被实例化,他认为 Restaurant 过于依赖 Chef 了,这不是废话吗?餐厅当然依赖与厨师了,一瞬间他灵机一动说:这种依赖关系是不可避免的,这样做是过度耦合的,我需要给他们解耦!Chef 的控制权并不应该由餐厅掌控,而应该是需要的时候由第三方空降,于是他把Chef 变成了 Restaurant 类的一个变量,Restaurant 不具备实例化他的能力,而是由第三方实例化好后直接赋值给Chef变量,于是他改写成如下代码:
public class Restaurant {private String name;private String specialty;private Chef chef;#=====================================public void setChef(Chef chef) {this.chef = chef;} #=====================================Restaurant(String name, String specialty) {this.name = name;this.specialty = specialty;}public void kitchen(){ #======================chef = 第三方力量直接空投 ;#======================System.out.println("我们的厨师长是:"+chef.name+",今年高龄:"+chef.age);chef.cook(); //厨师做菜}
}
如上面代码所示,第三方力量直接空投怎么实现呢?海绵宝宝思考了一夜,他想到了注射器,能不能像打针一样,把已经实例化好的Chef直接注射到 "chef = 第三方力量直接空投 ;"中的chef 呢?经过漫长的研究,海绵宝宝研发了一款框架,这种框架就是用来解决此问题,因为这是今年春天开始经历的困难,于是他把该框架命名成Spring。
3.1 Spring 依赖注入
他感悟道:如果一个类A 的功能实现需要借助于类B,那么就称类B是类A的依赖,如果在类A的内部去实例化类B, 那么两者之间会出现较高的耦合,一旦类B出现了问题,类A也需要进行改造,如果这样的情况较多,每个 类之间都有很多依赖,那么就会出现牵一发而动全身的情况,程序会极难维护,并且很容易出现问题。要解决这个问题,就要把A类对B类的控制权抽离出来,交给一个第三方去做,把控制权反转给第三方,就称 作控制反转(IOC Inversion Of Control)。控制反转是一种思想,是能够解决问题的一种可能的结果, 而依赖注入(Dependency Injection)就是其最典型的实现方法。由第三方(我们称作IOC容器)来控制依赖,把他通过构造函数 、属性或者工厂模式 等方法,注入到类A内,这样就极大程度的对类A和类B 进行了解耦。
上面 "chef = 第三方力量直接空投 ;" 写成:
ApplicationContext applicationContext = newClassPathXmlApplicationContext("applicationContext.xml");Chef chef = (Chef) applicationContext.getBean("chef");
<bean id="chef" class="com.myjava.zhangChef"/>
这一步还不是依赖注入,只是使用Spring的API实例化对象。因为有setChef 方法,我们可以配置Spring容器调用set方法进行注入,如下才是依赖注入,Spring会自动调用setChef方法给chef设置实例对象,因此不需要上面那一句“chef = 第三方力量直接空投”。真正的空投看不见!
<bean id="chef" class="com.myjava.LiChef"/>
<bean id="restaurant" class="com.myjava.Restaurant"><property name="chef" ref="chef"/>
</bean>
如果A类需要使用B类,那么我们称为B是A的依赖,我们需要Spring容器(第三方)将依赖B的示例注射给A。因此,需要在Spring核心配置文件中声明控制权,将指定全限定名称的类的控制权(创建权)交给Spring管理,意味着我们可以通过Spring的getBean()实例化。如果类 chef 后期需要升级维护,我们只需要更改其中的代码或者指定其他的类全限定名称(class名称)。
3.2 Bean 的依赖注入方式有两种
①构造方法
创建有参构造
public class UserServiceImpl implements Userservice{private UserDao userDao;public UserServiceImpl(UserDao userDao){this.userDao = userDao;}
}
配置Spring容器调用有参构造时进行注入
<bean id="userDao" class="com.myjava.dao.impl.UserDaoImpl"/>
<bean id="userService" class="com.myjava.service.impl.UserServiceImpl"> <constructor-arg name="userDao" ref="userDao"></constructor-arg>
</bean>
②set方法
在UserServiceImpl中添加setUserDao方法
public class UserServiceImpl implements UserService {private UserDao userDao;public void setUserDao(UserDao userDao) {this.userDao = userDao; } @Override public void save() { userDao.save();}
}
配置Spring容器调用set方法进行注入
<bean id="userDao" class="com.myjava.dao.impl.UserDaoImpl"/>
<bean id="userService" class="com.myjava.service.impl.UserServiceImpl"><property name="userDao" ref="userDao"/>
</bean>
4. 总结
依赖注入是反转控制的一种实现方法,我个人认为依赖外置比依赖注入更好理解,当然我可能理解的不太深刻有错误。因为原来的程序是直接在A里面实例化B,如果B变了那将导致很多业务代码需要改,这种依赖关系又是无法消除的,只能减弱这种依赖。那么,可以考虑将依赖写到配置文件中,需要的时候注入即可,这里将依赖写到配置文件可以说是依赖外置,但是Spring在运行时是实例化对象然后通过Set或者构造方法将其注入到A类中,因此,这是一个相对的说法。从A类角度出发是外置,从Spring角度是注入。以上内容参考了知乎胡小国博主的文章,感谢!以上内容有错误希望指正!
相关文章:
Spring依赖注入与反转控制到底是个啥?
目录 1. 引言 2. 管中窥豹 3.1 Spring 依赖注入 3.2 Bean 的依赖注入方式有两种 4. 总结 1. 引言 此文目的是用通俗易懂的语言讲清楚什么是依赖注入与反转控制,在看了大量的博客文章后归纳总结,便于后续巩固!我相信,大多数…...

Linux Shell脚本讲解
目录 Shell脚本基础 Shell脚本组成 Shell脚本工作方式 编写简单的Shell脚本 Shell脚本参数 Shell脚本接收参数 Shell脚本判断用户参数 文件测试与逻辑测试语句 整数测试比较语句 字符串比较语句 Shell流程控制 if条件判断语句 单分支 双分支 多分支 for循环语句…...

Linux:用户空间非法指针coredump简析
1. 前言 限于作者能力水平,本文可能存在谬误,因此而给读者带来的损失,作者不做任何承诺。 2. 背景 本文分析基于 ARM32 架构,Linux-4.14 内核代码。 3. 问题分析 3.1 测试范例 void main(void) {*(int *)0 8; }运行程序会 …...

带你玩转Jetson之Deepstream简明教程(四)DeepstreamApp如何使用以及用于工程验证。
1.DeepstreamApp是什么? 如果你安装完毕deepstream整体框架,会在你的系统执行目录内有可执行文件,文件名字是deepstream-app。这是一个可执行脚本文件,通过deepstream框架中的代码在安装的时候编译后install到系统根目录内。 此脚…...

快速搭建个人在线书库,随时随地畅享阅读!
前边我们利用NAS部署了个人的导航页、小说站、云笔记,今天,我们再看看怎么部署一个个人的在线书库。 相信很多朋友都在自己的电脑中收藏了大量的PDF、MOBI等格式的电子书籍,但是一旦换了一台设备,要么是无法翻阅,要么…...

电子纸墨水屏的现实应用场景
电子纸挺好个东西,大家都把注意力集中在商超场景 其实还有更多有趣的场景方案可用,价值也不小,比如: 一、仓库场景 通过亮灯拣选,提高仓库作业效率 二、仓库循环使用标签 做NFC类发卡式应用,替代传统纸…...

常量const、引用、指针的大杂烩
文章目录1 普通引用1.1 对普通值的普通引用1.2 对常量值的普通引用1.3 对普通指针的普通引用1.4 对常量指针的普通引用1.5 对指针常量的普通引用1.6 对指向常量的指针常量的普通引用2 常量引用2.1 对普通值的常量引用2.2 对常量值的常量引用2.3 对普通指针的常量引用2.4 对常量…...

宝塔搭建实战php开源likeadmin通用管理移动端uniapp源码(四)
大家好啊,我是测评君,欢迎来到web测评。 上一期给大家分享了pc端的部署方式,今天来给大家分享uniapp端在本地搭建,与打包发布到宝塔的方法。感兴趣的朋友可以自行下载学习。 技术架构 vscode node16 vue3 uniapp vite types…...
Hive的分区表与分桶表内部表外部表
文章目录1 Hive分区表1.1 Hive分区表的概念?1.1.1 分区表注意事项1.2 分区表物理存储结构1.3 分区表使用场景1.4 静态分区表是什么?1.4.1 静态分区表案例1.4.2 分区表练习一1.4.3 分区操作1.5 动态分区表是什么?1.5.1 动态态分区表案例&#…...

和数集团打造《神念无界:源起山海》,诠释链游领域创新与责任
首先,根据网上资料显示,一部《传奇》,二十年热血依旧。 《传奇》所缔造的成绩,承载的是多少人的青春回忆,《传奇》无疑已经在游戏史上写下了浓墨重彩的一笔。 相比《传奇》及背后的研发运营公司娱美德名声大噪&#x…...

小白入门模拟IC设计,如何快速学习?
众所周知,模拟电路很难学。以最普遍的晶体管来说,我们分析它的时候必须首先分析直流偏置,其次在分析交流输出电压。可以说,确定工作点就是一项相当麻烦的工作(实际中来说),晶体管的参数多、参数…...

51单片机——中断系统之外部中断实验,小白讲解,相互学习
中断介绍 中断是为使单片机具有对外部或内部随机发生的事件实时处理而设置的,中断功能的存在,很大程度上提高了单片机处理外部或内部事件的能力。它也是单片机最重要的功能之一,是我们学些单片机必须要掌握的。 为了更容易的理解中断概念&…...

如何设计一个秒杀系统
秒杀系统要如何设计? 前言 高并发下如何设计秒杀系统?这是一个高频面试题。这个问题看似简单,但是里面的水很深,它考查的是高并发场景下,从前端到后端多方面的知识。 秒杀一般出现在商城的促销活动中,指定…...
厄瓜多尔公司注册方案
简介: 经济概况与商机 厄瓜多尔是世界上第74大国家,是南美西部国家,与哥伦比亚,秘鲁和太平洋接壤。厄瓜多尔地处世界中心,地理位置优越,地理位置优越-赤道线零纬度,使其成为通往太平洋的理想枢…...

安全渗透环境准备(工具下载)
数据来源 01 一些VM虚拟机的安装 攻击机kali: kali官网 渗透测试工具Kali Linux安装与使用 kali汉化 虚拟机网络建议设置成NAT模式,桥接有时不稳定。 靶机OWASP_Broken_Web_Apps: 迅雷下载 网盘下载 安装教程 开机之后需要登录&am…...

118.(leaflet篇)leaflet空间判断-点与geojson面图层的空间关系(turf实现)
听老人家说:多看美女会长寿 地图之家总目录(订阅之前建议先查看该博客) 文章末尾处提供保证可运行完整代码包,运行如有问题,可“私信”博主。 效果如下所示: 下面献上完整代码,代码重要位置会做相应解释 <!DOCTYPE html> <html>...

目标检测与目标跟踪算法技术汇总
现如今chatgpt的爆火,我也使用了一段时间,问了许多关于人工智能技术的问题,基本是它能够回答了大部分的原理的,至于其人工智能涉及到的算法以及网络,考虑到也没有图,可能在给出这类回答上,是不太…...
Linux 系统启动过程
过去几十年,公用事业行业发生了重大变化。能源需求的转变导致企业利润率的波动,但不是运营成本的波动。 许多公用事业公司通过后勤部门流程自动化来削减成本,比如招采流程自动化。 在招采活动中,人工招采会产生盲点。由于公共事业…...
【每日一题Day118】LC1124表现良好的最长时间段 | 前缀和+单调栈/哈希表
表现良好的最长时间段【LC1124】 给你一份工作时间表 hours,上面记录着某一位员工每天的工作小时数。 我们认为当员工一天中的工作小时数大于 8 小时的时候,那么这一天就是「劳累的一天」。 所谓「表现良好的时间段」,意味在这段时间内&#…...
vue使用nprogress(进度条)
目录 1.安装 2.引入 3.配置 4.使用 5.使用场景 6.改变颜色 1.安装 npm install --save nprogress2.引入 import NProgress from nprogress import nprogress/nprogress.css3.配置 NProgress.configure({easing: ease, // 动画方式,和css动画属性一样&#…...

华为云AI开发平台ModelArts
华为云ModelArts:重塑AI开发流程的“智能引擎”与“创新加速器”! 在人工智能浪潮席卷全球的2025年,企业拥抱AI的意愿空前高涨,但技术门槛高、流程复杂、资源投入巨大的现实,却让许多创新构想止步于实验室。数据科学家…...

UE5 学习系列(三)创建和移动物体
这篇博客是该系列的第三篇,是在之前两篇博客的基础上展开,主要介绍如何在操作界面中创建和拖动物体,这篇博客跟随的视频链接如下: B 站视频:s03-创建和移动物体 如果你不打算开之前的博客并且对UE5 比较熟的话按照以…...

前端导出带有合并单元格的列表
// 导出async function exportExcel(fileName "共识调整.xlsx") {// 所有数据const exportData await getAllMainData();// 表头内容let fitstTitleList [];const secondTitleList [];allColumns.value.forEach(column > {if (!column.children) {fitstTitleL…...
Element Plus 表单(el-form)中关于正整数输入的校验规则
目录 1 单个正整数输入1.1 模板1.2 校验规则 2 两个正整数输入(联动)2.1 模板2.2 校验规则2.3 CSS 1 单个正整数输入 1.1 模板 <el-formref"formRef":model"formData":rules"formRules"label-width"150px"…...

保姆级教程:在无网络无显卡的Windows电脑的vscode本地部署deepseek
文章目录 1 前言2 部署流程2.1 准备工作2.2 Ollama2.2.1 使用有网络的电脑下载Ollama2.2.2 安装Ollama(有网络的电脑)2.2.3 安装Ollama(无网络的电脑)2.2.4 安装验证2.2.5 修改大模型安装位置2.2.6 下载Deepseek模型 2.3 将deepse…...

基于Java+VUE+MariaDB实现(Web)仿小米商城
仿小米商城 环境安装 nodejs maven JDK11 运行 mvn clean install -DskipTestscd adminmvn spring-boot:runcd ../webmvn spring-boot:runcd ../xiaomi-store-admin-vuenpm installnpm run servecd ../xiaomi-store-vuenpm installnpm run serve 注意:运行前…...

何谓AI编程【02】AI编程官网以优雅草星云智控为例建设实践-完善顶部-建立各项子页-调整排版-优雅草卓伊凡
何谓AI编程【02】AI编程官网以优雅草星云智控为例建设实践-完善顶部-建立各项子页-调整排版-优雅草卓伊凡 背景 我们以建设星云智控官网来做AI编程实践,很多人以为AI已经强大到不需要程序员了,其实不是,AI更加需要程序员,普通人…...
LangChain【6】之输出解析器:结构化LLM响应的关键工具
文章目录 一 LangChain输出解析器概述1.1 什么是输出解析器?1.2 主要功能与工作原理1.3 常用解析器类型 二 主要输出解析器类型2.1 Pydantic/Json输出解析器2.2 结构化输出解析器2.3 列表解析器2.4 日期解析器2.5 Json输出解析器2.6 xml输出解析器 三 高级使用技巧3…...
【深尚想】TPS54618CQRTERQ1汽车级同步降压转换器电源芯片全面解析
1. 元器件定义与技术特点 TPS54618CQRTERQ1 是德州仪器(TI)推出的一款 汽车级同步降压转换器(DC-DC开关稳压器),属于高性能电源管理芯片。核心特性包括: 输入电压范围:2.95V–6V,输…...

EasyRTC音视频实时通话功能在WebRTC与智能硬件整合中的应用与优势
一、WebRTC与智能硬件整合趋势 随着物联网和实时通信需求的爆发式增长,WebRTC作为开源实时通信技术,为浏览器与移动应用提供免插件的音视频通信能力,在智能硬件领域的融合应用已成必然趋势。智能硬件不再局限于单一功能,对实时…...