1. 什么是I0C/DI?
I0C (Inversion of Control)控制反转:所谓控制反转,就是把原先我们代码里面需要实现的对象创建、依赖的代码,反转给容器来帮忙实现。那么必然的我们需要创建一个容器, 同时需要一种 描述来让容器知道需要创建的对象与对象的关系。这个描述最具体表现就是我们可配置的文件。
DI (Dependency Injection)依赖注入:就是指对象是被动接受依赖类而不是自己主动去找,换句话说就
是指对象不是从容器中查找它依赖的类,而是在容器实例化对象的时候主动将它依赖的类注入给它。
先从我们自己设计这样-一个视角来考虑:
对象和对象关系怎么表示?
可以用xml,properties文件 等语义化配置文件表示。
描述对象关系的文件存放在哪里?
可能是classpath,filesystem, 或者是URL 网络资源,servletContext等。
回到正题,有了配置文件,还需要对配置文件解析。
不同的配置文件对对象的描述不一样,如标准的,自定义声明式的,如何统-一?在内部需要有一个统一的关于对象的定义,所有外部的描述都必须转化成统一的描述定义。
如何对不同的配置文件进行解析?需要对不同的配置文件语法,采用不同的解析器
2. Spring I0C 体系结构
2.1 BeanFactory
Spring Bean的创建是典型的工厂模式,这一系列的Bean工厂,也即I0C容器为开发者管理对象间的依赖关系提供了很多便利和基础服务,在Spring中有许多的I0C容器的实现供用户选择和使用,其相互关系如下:
其中BeanFactory作为最顶层的一个接口类,它定义了IOC容器的基本功能规范,BeanFactory 有三个子类: ListableBeanFactory、HierarchicalBeanFactory和AutowireCapableBeanFactory。但是从上图中我们可以发现最终的默认实现类是DefaultListableBeanFactory, 他实现了所有的接口。那为何要定义这么多层次的接口呢? 查阅这些接口的源码和说明发现,每个接口都有他使用的场合,它主要是为了区分在Spring内部在操作过程中对象的传递和转化过程中,对对象的数据访问所做的限制。例如ListableBeanFactory 接口表示这些Bean 是可列表的, 而 HierarchicalBeanFactory表示的是这些 Bean是有继承关系的,也就是每个Bean有可 能有父 Bean。AutowireCapableBeanFactory接口定义Bean 的自动装配规则。这四个接口共同定义了Bean 的集合、Bean之间的关系、 以及Bean行为。
最基本的IOC容器接口:BeanFactory
public interface BeanFactory {
// 对FactoryBean的转义定义,因为如果使用bean的名字检索FactoryBean得到的对象是工厂生成的对象,//如果需要得到工厂本身,需要转义
String FACTORY_BEAN_PREFIX = "&";
// 根据bean的名字,获取在IOC容器中得到bean实例
Object getBean(String name) throws BeansException;
// 根据bean的名字和Class类型来得到bean实例,增加了类型安全验证机制。
Object getBean(String name, Class reguiredType) throws BeansException;
// 提供对bean的检索,看看是否在IOC容器有这个名字的bean
boolean containsBean(String name);
// 根据bean名字得到bean实例,并同时判断这个bean是不是单例
boolean isSingleton(String name) throws NoSuchBeanDef initi onException;
// 得到bean实例的Class类型
Class getType(String name) throws NoSuchBeanDefinitionException;
// 得到bean的别名,如果根据别名检索,那么其原名也会被检索出来
String[] getAliases(String name);
}
在BeanFactory里只对I0C容器的基本行为作了定义,根本不关心你的bean是如何定义怎样加载的。正如我们只关心工厂里得到什么的产品对象,至于工厂是怎么生产这些对象的,这个基本的接口不关心。
而要知道工厂是如何产生对象的,我们需要看具体的IOC容器实现,Spring 提供了许多I0C容器的实现。比如XmlBeanFactory, ClasspathXmlApplicationContext等。其中XmlBeanFactory就是针对最基本的I0C容器的实现,这个I0C容器可以读取XML文件定义的BeanDefinition (XML文件中对bean的描述) , 如果说XmlBeanFactory是容器中的属丝,ApplicationContext应该算容器中的高帅富.
Applicat ionContext是Spring提供的一个高级的IOC容器,它除了能够提供IOC容器的基本功能外,还为用户提供了以下的附加服务。
从ApplicationContext接口的实现,我们看出其特点:
- 支持信息源, 可以实现国际化。( 实现MessageSource接口)
- 访问资源。 (实现ResourcePatternResolver接口,这个后面要讲)
- 支持应用事件。 (实现ApplicationEventPublisher接口)
2.2 BeanDefinition(一种规范)
springIOC容器管理了我们定义的各种bean对象及其相互的关系,bean对象在Spring实现中是以BeanDefinition来描述的,器继承关系如下:
Bean的解析过程非常复杂,功能被分的很细,因为这里需要被扩展的地方很多,必须保证有足够的灵活性,一应对可能的变化。Bean的解析主要就是对spring配置文件的解析。这个解析过程主要是通过下图中的类完成:
2.3 I0C容器的初始化
IOC容器的初始化包括BeanDefinition的Resource定位、载入和注册这三个基本的过程。我们以ApplicationContext为例讲解,ApplicationContext系列容器也许是我们最熟悉的,因为web项目中使用的XmlWebApplicationContext就属于这个继承体系,还有ClasspathXmlApplicationContext等,其继承体系如下图所示:
ApplicationContext允许上下文嵌套,通过保持父上下文可以维持一个 上下文体系。对于bean的查找可以在这个上下文体系中发生,首先检查当前上下文,其次是父上下文,逐级向上,这样为不同的Spring 应用提供了一个共享的bean定义环境。
下面我们分别简单地演示一下两种ioc容器的创建过程
2.3.1 XmlBeanFactory(属丝I0C)的整个流程
通过XmlBeanFactory的源码, 我们发现:
public class XmlBeanFactory extends DefaultListableBeanFactory{
private final XmlBeanDefinitionReader reader;
public XmlBeanFactory(Resource resource )throws BeansException{
this(resource, null);
}
public XmlBeanFactory(Resource resource, BeanFactory parentBeanFactory) throws BeansException{
super (parentBeanFactory) ;
this.reader = new XmlBeanDefinitionReader(this);
this.reader.loadBeanDefinitions(resource);
}
}
调用全过程还原,定位、载入、注册
//根据Xml配置文件创建Resource资源对象,该对象中包含了BeanDefinition的信息
ClassPathResource resource = new ClassPathResource( "application-context.xml" );
//创建DefaultListableBeanFactory
DefaultListableBeanFactory factory = new DefaultlistableBeanFactory();
//创建XmlBeanDefinitionReader读取器,用于载入BeanDefinition。之所以需要BeanFactory作为参数,是因为会将读取的信息回调配置给factory
XmlBeanDefinitionReader reader =new XmlBeanDefinitionReader( factory);
//XmlBeanDef initionReader执行载入BeanDefinition的方法,最后会完成Bean的载入和注册。完成后Bean就成功的放置到IOC容器当中,以后我们就可以从中取得Bean来使用
reader.loadBeanDefinitions(resource);
通过前面的源码,this.reder= new XmlBeanDefinitionReader(this); 中其中的this传的就是factory对象。
2.3.2 FileSystemXmlApplicationContext的IOC容器过程
1.高富帅IOC解剖
ApplicationContext =new FileSystemXmlApplicationContext(xmlPath)
先看构造函数:
调用构造函数: