目录
1.Spring 基础
2.传统程序开发
3.IoC程序开发(解耦)
4.DI
4.1 IoC 和 DI 有什么区别
Spring 指的是 Spring Framework(Spring 框架),它是⼀个开源框架,有着活跃⽽庞⼤的社区,这就是它之所以能⻓久不衰的原因。Spring ⽀持⼴泛的应⽤场景,它可以让 Java 企业级的应⽤程序开发起来更简单。
一句话概括:Spring 是包含了众多工具方法的 IoC 容器
容器:⽤来容纳某种物品的(基本)装置(例如:List/Map ---> 数据存储容器、Tomcat ----> Web 容器)
IoC:IoC = Inversion of Control 翻译成中⽂是“控制反转”的意思,也就是说 Spring 是⼀个“控制反转”的容器
控制(权)反转:对象的生命周期,不是由程序员(或当前代码片段)来控制,而是由 Spring(Spring 容器/ IoC容器)来控制
IoC最大的优点:可以实现解耦(松耦合)
假如,我们现在构建⼀辆“⻋”的程序,我们的实现思路是这样的:
/**
* 传统开发:汽车对象
*/
public class Car {
private Framework framework;//依赖车声,建立车身
//在构造方法中 new
public Car() {
this.framework = new Framework();
}
public void init() {
//依赖车身
System.out.println("执行了 init 方法");
framework.init();
}
}
/**
* 车身类
*/
public class Framework {
private Bottom bottom;
public Framework() {
this.bottom = new Bottom();
}
public void init() {
//依赖底盘
System.out.println("执行了 Framework init 方法");
bottom.init();
}
}
/**
* 底盘类
*/
public class Bottom {
private Tire tire;
public Bottom() {
this.tire = new Tire();
}
public void init() {
//依赖轮胎
System.out.println("执行了 Bottom init 方法");
tire.init();
}
}
/**
* 轮胎类
*/
public class Tire {
private int size = 15;
public void init() {
System.out.println("执行了 Tire init,Size:"+ size);
}
}
这个时候我们执行这段代码:
public class Test {
public static void main(String[] args) {
Car car = new Car();
car.init();
}
}
上述代码中轮胎的尺寸是固定的,然而随着对的车的需求量越来越⼤,个性化需求也会越来越多,这 时候我们就需要加工多种尺⼨的轮胎,那这个时候就要对上面的程序进行修改:
注意每一类中size的改动
/**
* 传统开发:汽车对象
*/
public class Car {
private Framework framework;
public Car(int size) {
this.framework = new Framework(size);
}
public void init() {
//依赖车身
System.out.println("执行了 init 方法");
framework.init();
}
}
/**
* 车身类
*/
public class Framework {
private Bottom bottom;
public Framework(int size) {
this.bottom = new Bottom(size);
}
public void init() {
//以来底盘
System.out.println("执行了 Framework init 方法");
bottom.init();
}
}
/**
* 底盘类
*/
public class Bottom {
private Tire tire;
//Botton 中没有size参数,继续让别人传参数
public Bottom(int size) {
this.tire = new Tire(size);
}
public void init() {
//依赖轮胎
System.out.println("执行了 Bottom init 方法");
tire.init();
}
}
/**
* 轮胎类
*/
public class Tire {
private int size = 15;
//传递 size 参数
public Tire(int size) {
this.size = size;
}
public void init() {
System.out.println("执行了 Tire init,Size:"+ size);
}
}
public class Test {
public static void main(String[] args) {
Car car = new Car(20);
car.init();
}
}
这个时候我们发现,虽然我们可以改变轮胎的尺寸,但是最底层代码改动之后,整个调用链上的所有代码都需要修改——耦合性问题
需求指挥越来越多,改动的也会越来越多,这个时候我们就可以使用 IoC 进行解耦
/**
* IoC:汽车对象
*/
public class Car {
private Framework framework;
//初始化(不是用传统开发):让框架传入(不管咋样传入),声明使用Framework
//如果是一个IoC框架,运行代码,会拿到当前框架中的Framework 赋值到当前变量中就会有 framework
public Car(Framework framework) {
this.framework = framework;
}
public void init() {
System.out.println("Car init");
framework.init();
}
}
public class Framework {
private Bottom bottom;
public Framework(Bottom bottom) {
this.bottom = bottom;
}
public void init() {
System.out.println("Framework init");
bottom.init();
}
}
public class Bottom {
private Tire tire;
public Bottom(Tire tire) {
this.tire = tire;
}
public void init() {
System.out.println("Bottom init");
tire.init();
}
}
public class Tire {
private int size = 15;
public Tire() {
}
public void init() {
System.out.println("Tire init, Size:" + size);
}
}
/**
* 模拟 IoC 容器
*/
public class Test {
private Tire tire;
private Bottom bottom;
private Framework framework;
private Car car;
public Test() {
this.tire = new Tire();
this.bottom = new Bottom(this.tire);
this.framework = new Framework(this.bottom);
this.car = new Car(this.framework);
}
public static void main(String[] args) {
Test test = new Test();
test.car.init();
}
}
需要改尺寸只需要修改 Tire 类中的尺寸:
public class Tire {
private int size = 15;
public Tire(int size) {
this.size = size;
}
public void init() {
System.out.println("Tire init, Size:" + size);
}
}
这个时候只需要改 IoC 容器(Spring 框架 )即可(IoC 与用户没有关系),Spring 框架会自动设置参数
public class Test {
private Tire tire;
private Bottom bottom;
private Framework framework;
private Car car;
public Test() {
this.tire = new Tire(20);
this.bottom = new Bottom(this.tire);
this.framework = new Framework(this.bottom);
this.car = new Car(this.framework);
}
public static void main(String[] args) {
Test test = new Test();
test.car.init();
}
}
代码经过以上调整,⽆论底层类如何变化,整个调⽤链是不⽤做任何改变的,这样就完成了代码之间的解耦,从⽽实现了更加灵活、通⽤的程序设计了
通⽤程序的实现代码,类的创建顺序是反的,传统代码是 Car 控制并创建了Framework,Framework 创建并创建了 Bottom,依次往下,⽽改进之后的控制权发⽣的反转,不再是上级对象创建并控制下级对象了,⽽是下级对象把注⼊将当前对象中,下级的控制权不再由上级类控制了,这样即使下级类发⽣任何改变,当前类都是不受影响的,这就是典型的控制反转,也就是 IoC 的实现思想。
既然 Spring 是⼀个 IoC(控制反转)容器,重点还在“容器”⼆字上,那么它就具备两个最基础的功能:
也就是说学 Spring 最核心的功能,就是学如何将对象存入到 Spring 中,再从 Spring 中获取对象的过程
将对象存放到容器中的好处:将对象存储在 IoC 容器相当于将以后可能⽤的所有共具制作好都放到仓库中,需要的时候直接取就行了,用完再把它放回到仓库。而new 对象的方式相当于,每次需要工具了,才现做,用完就扔掉了也不会保存,下次再⽤的时候还得重新做,这就是 IoC 容器和普通程序开发的区别
DI 是 Dependency Injection 的缩写,翻译成中文是“依赖注入”的意思:指的是由 IoC 容器在运行期间,动态的将依赖对象获取到的过程
public class Car {
private Framework framework;
public Car(Framework framework) {
this.framework = framework;
}
public void init() {
System.out.println("Car init");
framework.init();
}
}
在运行 Car 时,动态的将 Framework 拿到当前类中的过程就叫做 依赖注入 (在运行类的时候,需要有一个依赖类,依赖类就会从容器中动态的查询出来,赋值给变量,就可以直接使用这个方法)
所以,依赖注⼊(DI)和控制反转(IoC)是从不同的⻆度的描述的同⼀件事情,就是指通过引⼊ IoC 容器,利⽤依赖关系注⼊的⽅式,实现对象之间的解耦;
IoC 是“目标”也是⼀种思想,而目标和思想只是⼀种指导原则,最终还是要有可行的落地方案,而 DI 就属于具体的实现。
比如说我今天吃⼀顿美餐,那么“美餐”是目标(是 IoC),但最后我是吃火锅还是烧烤?这就是具体的实现,就是DI
IoC 和 DI 都是 Spring 框架中的核心概念,它们的区别在于:
简单来说,它们的关系是:
所以 IoC 是更基础和广义的概念,DI 可以说是 IoC 的一种实现手段。大多数情况下,我们提到 IoC 的时候,其实意味着 DI,因为 DI 已经是 IoC 最常见和广泛使用的实现方式了。
例如在 Spring 框架中:
- IoC 体现为 Spring 容器承担了对象创建及依赖关系管理的控制权。
- DI 体现为 Spring 容器通过构造方法注入、Setter 方法注入等方式,将依赖对象注入到需要依赖的对象中。
所以综上,IoC 和 DI 之间的关系可以这样理解:
- IoC 是理论,DI 是实践。
- IoC 是思想,DI 是手段。
- IoC 是整体,DI 是部分。