- Spring 是什么?
- 一、什么是容器?
- 二、什么是 IoC?
- 2.1 传统程序开发
- 2.2 控制反转式程序开发
- 2.3 对比总结规律
- 三、理解 Spring IoC
- 四、DI 概念说明
Spring 是什么?
我们通常说的 Spring 是指 Spring Framework(Spring 框架),它是一个开源的框架,有着活跃而庞大的社区,这就是它长久不衰的原因。
Spring 支持广泛的应用场景,它可以让 Java 企业级的应用程序开发起来更加简单。
用一句话概括 Spring:Spring 是包含了众多工具方法的 IoC 容器。
一、什么是容器?
容器是用来容纳某种物品的基本装置。——来自:百度百科
Java 中的容器有哪些?例如:
List / Map -> 数据存储容器
Tomcat -> Web 容器
二、什么是 IoC?
Spring 也是一个容器,Spring 是什么容器呢?Spring 是一个 IoC 容器。
什么是 IoC?
IoC = Inversion of Control 翻译成中文是 “控制反转” 的意思,也就是说 Spring 是一个 “控制反转” 的容器。
什么是 “控制反转” 容器?我将举例演示一下。
2.1 传统程序开发
我们现在要构建一辆 “车” 的程序,我们的实现思路是这样的:
构建一辆车(Car),车需要依赖车身(FrameWork),而车身需要依赖底盘(Bottom),而底盘需要依赖轮胎(Tire),最终程序的实现代码如下:
public static void main(String[] args) {
Car car = new Car();
car.init();
}
//汽车对象
static class Car{
public void init(){
Framework framework = new Framework();
framework.init();
}
}
//车身类
static class Framework{
public void init(){
Bottom bottom = new Bottom();
bottom.init();
}
}
//底盘类
static class Bottom{
public void init(){
Tire tire = new Tire();
tire.init();
}
}
//轮胎类
static class Tire{
private int size = 20;
public void init(){
System.out.println("轮胎尺寸:" + size);
}
}
传统程序开发的缺陷
以上程序中,轮胎的尺寸是固定的。然而在现实生活中,对汽车的需求量越来越大,个性化也会越来越多,不可能一直使用同一种尺寸的轮胎。
这时候就需要对上面的程序进行修改了…
public static void main(String[] args) {
Car car = new Car(18);
car.init();
}
//汽车对象
static class Car{
private Framework framework;
public Car(int size){
framework = new Framework(size);
}
public void init(){
framework.init();
}
}
//车身类
static class Framework{
private Bottom bottom;
public Framework(int size){
bottom = new Bottom(size);
}
public void init(){
bottom.init();
}
}
//底盘类
static class Bottom{
private Tire tire;
public Bottom(int size) {
tire = new Tire(size);
}
public void init(){
tire.init();
}
}
//轮胎类
static class Tire{
private int size;
public Tire(int size){
this.size = size;
}
public void init(){
System.out.println("轮胎尺寸:" + size);
}
}
显然这样的修改,付出的代价是蛮大的。最底层代码改动后,整条调用链上的所有代码都需要修改。
如何解决上述的问题呢?
我们可以尝试不在每个类中自己创建下级类,如果自己创建下级类就会出现当下级类发生改变操作,自己也要跟着修改。
此时,我们只需要将原来由自己创建的下级类,改为传递的方式(也称为注入的方式)。因为我们不需要在当前类中创建下级类了,所以下级类即使发生改变(创建或者减少参数),当前类本身也无需修改如何代码,这样就完成了程序的解耦。
就例如:
我们创造一辆完整的汽车,如果所有的配件都是自己打造,那么当客户需求发生改变的时候,轮胎尺寸改为 21 寸,车漆改为欧泊银,这时候我们只能再次手动来改。但如果我们是把轮胎外包出去,那么即使轮胎的需求发生改变,我们只需要向代理工厂下订单就行了,我们自身是不需要出力的。
2.2 控制反转式程序开发
基于以上的思路,我们把调用汽车的程序示例修改一下,把创建子类的方式,改为注入传递的方式~
public static void main(String[] args) {
Tire tire = new Tire(20);
Bottom bottom = new Bottom(tire);
Framework framework = new Framework(bottom);
Car car = new Car(framework);
car.init();
}
//汽车对象
static class Car{
private Framework framework;
public Car(Framework framework){
this.framework = framework;
}
public void init(){
framework.init();
}
}
//车身类
static class Framework{
private Bottom bottom;
public Framework(Bottom bottom){
this.bottom = bottom;
}
public void init(){
bottom.init();
}
}
//底盘类
static class Bottom{
private Tire tire;
public Bottom(Tire tire) {
this.tire = tire;
}
public void init(){
tire.init();
}
}
//轮胎类
static class Tire{
private int size;
public Tire(int size){
this.size = size;
}
public void init(){
System.out.println("轮胎尺寸:" + size);
}
}
感觉各个类都不相互影响,只有到了 main 方法中,才把每个类提供的零件组装起来成为一辆汽车。
代码经过以上调整,无论底层类如何变化,整个调用链是不用做任何改变的,这样就完成了代码之间的解耦,从而实现了更加灵活、更加通用的程序设计了。
2.3 对比总结规律
传统代码中对象创建顺序是:Car -> Framework -> Bottom -> Tire
改进之后解耦的代码的对象创建顺序是:Tire -> Bottom -> Framework -> Car
传统代码是 Car 控制并创建了 Framework,Framework 控制并创建了 Bottom,依次往下。
而改进之后的控制权发生了反转,不再是上级对象创建并控制下级对象了,而是下级对象注入到当前对象中,下级的控制权不再是上级类掌控,这样即使下级类发生了改变,当前类和上级类都是不受影响的,这就是典型的控制反转,也就是 IoC 的实现思想。
三、理解 Spring IoC
回到我们一开始的主题 Spring,Spring 是包含了多个工具方法的 IoC容器,这就是对 Spring 最核心的总结。
如何理解 “Spring 是一个 IoC 容器” 这句话?
既然 Spring 是一个 IoC(控制反转)容器,重点还是在 “容器” 二字上,那么它就具备两个最基本的功能:
① 将对象存储到容器中;
② 从容器中取出对象。
也就是说学 Spring 最核心的功能,就是学会如何将对象存储到 Spring 中,再从 Spring 中获取对象的过程。
将对象存放到容器中的好处:
将对象存储到 IoC 容器相当于将以后可能用到所有工具制作好都放到仓库中,需要直接取,用完再把它放回仓库。
而 new 对象的方式相当于,每次需要工具,现场做,用完就直接扔掉了,下次再用的时候还得重新做。
这就是 IoC 容器和普通程序开发的区别。
Spring 是一个 IoC 容器,说的是对象的创建和销毁的权利都交给 Spring 来管理,它本身又具备了存储对象和获取对象的能力。
四、DI 概念说明
提到 IoC 不得不提另一个词,就是 “DI”,DI 是 Dependency Injection 的缩写,翻译成中文是 “依赖注入” 的意思。
所谓依赖注入,就是由 IoC 容器在运行期间,动态地将某种依赖关系注入到对象之中。所以,依赖注入(DI)和控制反转(IoC)是从不同的角度描述同一件事情,就是指通过引入 IoC 容器,利用依赖关系注入的方式,实现对象之间的解耦。
IoC 是一种思想或是一个目标,最终还是要有可行的落地方案,而 DI 就是属于具体的实现。 例如今天心情好,去喝杯奶茶(是思想和目标 IoC),最后去茶百道选了芋圆奶茶(是具体的实现 DI)。