1. 概念
首先,spring是一个框架,一个帮助我们快速开发web应用的框架。在说spring之前,我们先来谈谈web应用中,2个非常重要的概念
- IOC:全称叫Inversion of Control,翻译过来叫控制反转
- AOP:全称叫Aspect Oriented Programming,翻译过来叫面向切面编程
注意,这2个是软件开发中的设计思想,并不是spring独有的,许多文章把spring等同于IOC和AOP,真是误人子弟,IOC和AOP只是一种软件设计思想,并不是什么具体代码,有许多框架实现了这2个功能,像JBoss、HiveMind、EJB。
那么,我们今天来谈谈IOC。关于AOP请参考我的另一篇文章谈谈对Spring的理解之AOP
2. 控制反转是什么意思?
用一句话概括就是:将对象的创建、依赖过程交给容器管理,即把对象的控制全交给容器。相应的这个容器叫它IOC容器(有的框架也叫DI容器)。
其实,我们要使用一个对象,直接new就好了,为什么要这么干呢?这样做有什么好处?
2.1 new的方式创建对象
我们假设有一个对象A,需要用到对象B,那么由下面的代码
public class A {
public static void main(String[] args) {
// 直接通过new的方式
B b = new B("张三", 20);
}
}
这样的代码,好像看也没什么问题。但是由于在A里硬编码去new B,这样的代码是高度耦合的。不要小看耦合带来的破坏,当系统非常复杂的时候,当系统多个地方都去new B的时候,代码是非常不健壮的。
2.2 工厂设计模式解决创建对象的耦合问题
那么,第一种解决方案出来了,通过工厂设计模式来解决new对象的耦合问题。将使用的对象抽象出一个接口,然后通过调用工厂方法,传入标识,来获取具体的实现类(工厂设计模式可以参考我的另一篇文章:工厂设计模式介绍)
有如下代码:
// 简单工厂
public class SimpleExampleFactory {
public static IExample getExample(String flag) {
if ("B".equals(flag)) {
return new BExampleImpl();
} else if ("C".equals(flag)) {
return new CExampleImpl();
} else {
throw new RuntimeException("Unknown flag: " + flag);
}
}
}
统一通过调用工厂方法来获取对象。但是,这同样会带来工厂设计模式的缺点,即维护工厂类变的复杂,且工厂类里面也是硬编码创建对象的。
思考一下,如果工厂类里,创建对象的方法,是通过XML或者注解的方式来创建,是不是就大大的解耦了,恭喜你,其实spring的IOC容器简单说就是BeanFactory工厂,通过读取XML配置或者注解的方式来创建对象的
3. spring对IOC的实现
IOC模式,有2种实现的方式:
- DI:全称叫Dependency Injection,就是依赖注入
- DL:全称叫Dependency Lookup,就是依赖查找
而spring就是通过DI依赖注入的方式来实现IOC模式的,总的来说,有两个大步骤:
- 根据XML配置文件或者注解(如常用的@service、@control…)来创建对象,创建出一个一个的bean,并放到IOC容器中管理。
- 如果代码中引用了IOC容器中的对象(如注解@Autowire),那么将此对象引用指向IOC容器中的bean地址。
本篇文章只是说spring IOC的思想,那么这里只列出来伪代码实现过程。spring定义了一个接口BeanDefinition,接口里定义了创建bean的一些约束。第一步是将XML或者注解类,通过BeanDefinitionReader,构建成一个个的BeanDefinition
注意,bean的实例化和属性赋值,是分开的
在bean的实例化和引用其他bean对象的时候,会存在一个相互依赖的问题,spring通过三级缓存map结构解决bean对象之间的相互依赖,这里也不做讨论了。