简明依赖注入(Dependency Injection)
前言
这是因特奈特上面不知道第几万篇讲依赖注入(Dependency Injection)的文章,但是说明白的却寥寥无几,这篇文章尝试控制字数同时不做大多数。
首先,依赖注入的是一件很简单的事情。
为什么需要依赖注入
然后,假设我们有一个汽车Car,一个引擎接口Engine,两个引擎具体实现Level4Engine
,Level5Engine
。汽车可以长这样:
public class Car{
private Engine e;
public Car(){
e = new Level4Engine();
}
public void ignite(){
System.out.println()
}
}
现在要让汽车点火,简单:
public static void main(String[] args) {
Car c = new Car();
c.ignite();
}
但是假如我们想要换一个更高级的引擎,我们不得不修改Car
的构造函数:
~~ e = new Level4Engine();
~~
e = new Level5Engine();
然后重新编译。这就是代码的耦合,一方面假如需求不会经常改变,这个汽车只会使用Level4Engine
,那没问题,这个代码很完美。但另一方面,假如引擎有多个,需求会经常改变,我们发现Level4Engine
还不行,需要更高级的,而且新引擎还需要进行一系列复杂配置,那这个耦合就是灾难了。只是装配汽车的血汗工人,懂不了那么多的。
怎么进行依赖注入
依赖注入就是为了解决上述问题而生的。用依赖注入的写法解决上面的问题:
public class Car{
private Engine e;
public Car(Engine e){
this.e = e;
}
public void ignite(){
System.out.println()
}
}
// 也可以使用xml进行配置
@Confignuration
public CarFactory{
@Bean
public Engine engine(){
var e = new Level5Engine();
e.complexConfig();
return e;
}
@Bean
public Car car(Engine e){
return new Car(e);
}
}
这里Car对Engine的依赖被抽了出去。Car不负责创建Engine,也不负责/无能力配置Enging。那么Engine抽出到了哪?又由谁注入给Car?总不能让Car对着一个壳子(Engine接口)点火吧。
答案当然是spring。spring把它们抽象为Bean,每个@Bean
都通知spring
嘿我要给你一个新的bean,以后就交给你来管理了。
DI的优势
这样既解决了上述"汽车装配工需要引擎配置知识"的问题,也解决了"更改引擎非常困难"的问题:
- 引擎制造者只关注如何制造出引擎,当现在生产条件不成熟就提供
Level4Engine
,反之就提供Level5Engine
,可以随时更改并对其进行配置 - 汽车装配工只关注装配工作,而不需要配置引擎。
- 每次引擎更改后只需要对这个配置类进行编译,如果使用xml连编译也不需要了。
这真的就是依赖注入的全部内容了,不过围绕依赖注入相关还有很多话题可以讨论,下面扩展就是两个。
扩展1:使用自动装配代替手动装配
演示了在CarFactory
中手动car,还没完,spring还能更聪明一些,它可以通过自动装配完成这个配置工作:
@Component
public class Car{
private Engine e;
@Autowired
public Car(Engine e){
this.e = e;
}
public void ignite(){
System.out.println()
}
}
@Component
public class Level5Engine{
public void complexConfig(){
System.out.println("really complex stuff...");
}
}
@Confignuration
@ComponentScan
public class CarFactory{}
CarFactory
的@ComponentScan
告诉spring扫描当前类所在包下面的所有类,如果找到@Component
注解就加入spring bean容器。这里明显Car和Level5Engine加入了容器(默认会类名首字母小写,所以加入的是car
和level5Engine
)。然后@Autowired
在当前容器中查找,如果找到需要注入的类型就自动注入:
@Autowired
public Car(Engine e){
this.e = e;
}
Car的装配需要一个引擎,spring容器刚好有一个实现了Engine的Level5Engine引擎,所以这里自动注入。
扩展2: NoUniqueBeanDefinitionException自动装配歧义
最后一个不常见的问题,假如我们把两个引擎都标注了@Component
会怎么样:
@Component
public class Level5Engine{
}
@Component
public class Level4Engine{
}
spring不知道用哪一个注入给car,所以抛出NoUniqueBeanDefinitionException
,表示有多个候选注入对象,需要我们手动缩小范围(@Qualifier
,@Component value
,@Primary
),关于这部分内容可以参见其他文章。
简明依赖注入(Dependency Injection)的更多相关文章
- 控制反转Inversion of Control (IoC) 与 依赖注入Dependency Injection (DI)
控制反转和依赖注入 控制反转和依赖注入是两个密不可分的方法用来分离你应用程序中的依赖性.控制反转Inversion of Control (IoC) 意味着一个对象不会新创建一个对象并依赖着它来完成工 ...
- 14.AutoMapper 之依赖注入(Dependency Injection)
https://www.jianshu.com/p/f66447282780 依赖注入(Dependency Injection) AutoMapper支持使用静态服务定位构建自定义值解析器和自定 ...
- 依赖注入 | Dependency Injection
原文链接: Angular Dependency Injection翻译人员: 铁锚翻译时间: 2014年02月10日说明: 译者认为,本文中所有名词性的"依赖" 都可以理解为 & ...
- Spring点滴七:Spring中依赖注入(Dependency Injection:DI)
Spring机制中主要有两种依赖注入:Constructor-based Dependency Injection(基于构造方法依赖注入) 和 Setter-based Dependency Inje ...
- 设计模式之————依赖注入(Dependency Injection)与控制反转(Inversion of Controller)
参考链接: 依赖注入(DI) or 控制反转(IoC) laravel 学习笔记 —— 神奇的服务容器 PHP 依赖注入,从此不再考虑加载顺序 名词解释 IoC(Inversion of Contro ...
- 理解依赖注入(Dependency Injection)
理解依赖注入 Yii2.0 使用了依赖注入的思想.正是使用这种模式,使得Yii2异常灵活和强大.千万不要以为这是很玄乎的东西,看完下面的两个例子就懂了. class SessionStorage { ...
- AngularJS - 依赖注入(Dependency Injection)
点击查看AngularJS系列目录 转载请注明出处:http://www.cnblogs.com/leosx/ 依赖注入 依赖注入是软件设计模式中的一部分,用于处理组件是如何得到它说依赖的其它组件的. ...
- Spring之对象依赖关系(依赖注入Dependency Injection)
承接上篇: Spring中,如何给对象的属性赋值: 1:通过构造函数,如下所示: <!-- 1:构造函数赋初始值 --><bean id="user1" clas ...
- MVC使用StructureMap实现依赖注入Dependency Injection
使用StructureMap也可以实现在MVC中的依赖注入,为此,我们不仅要使用StructureMap注册各种接口及其实现,还需要自定义控制器工厂,借助StructureMap来生成controll ...
随机推荐
- Intellij IDEA 安装插件 报 ‘plugin xxxx is incompatible‘ 解决方案
网上下载安装LOMBOK失败,直接下载插件安装: 在离线安装IDEA插件的时候,可能会出现该问题.引起的原因主要就是版本号不一致. 下面介绍下离线安装找到合适的版本号. 1.在IDEA的help-&g ...
- 基于HTML5的RDP访问实战
基于HTML5的RDP访问实战 1.安装guacamole 2.下载源码 3.安装服务端 安装报错 错误 参考 http://www.remotespark.com/html5.html ...
- 7-set用法详解
C++中set用法详解 转载 http://blog.csdn.net/yas12345678/article/details/52601454 C++ / set 更详细见:http://www.c ...
- 关于C++中不同类之间的赋值问题——存疑
operator=不能重载为全局函数.理由如下 void operator=(int i , A& a) { a.a = i } ; 那么将会出现 99 = a 这种代码,但是99不是左值, ...
- OracleBulkCopy 修正帮
using System;using System.Collections.Generic;using System.Data;using System.Linq;using System.Refle ...
- 利用Ant与Proguard混淆
利用Ant与Proguard混淆 摘自:https://blog.csdn.net/forlong401/article/details/22956711 2014年04月04日 20:20:21 f ...
- C#调用COM组件遇到的问题及解决办法
1.无法嵌入互操作类型"xxx",请改用适用的接口. 解决办法: - 将所引用的程序集的[嵌入互操作类型]属性设置为[False]. 2.System.Runtime.Intero ...
- Go 语言并发笔记
前言: 本文是学习<<go语言程序设计>> -- 清华大学出版社(王鹏 编著) 的2014年1月第一版 做的一些笔记 , 如有侵权, 请告知笔者, 将在24小时内删除, 转载请 ...
- srand()、rand()、time()函数的用法
srand()就是给rand()提供种子seed. 如果srand每次输入的数值是一样的,那么每次运行产生的随机数也是一样的. 以一个固定的数值作为种子是一个缺点.通常的做法是 :以这样一句srand ...
- Java基本类型学习
基本类型,或者叫做内置类型,是JAVA中不同于类的特殊类型.它们是我们编程中使用最频繁的类型.java是一种强类型语言,第一次申明变量必须说明数据类型,第一次变量赋值称为变量的初始化. 1. Java ...