本篇内容包括:IOC 和 DI 的概念、Spring 容器,即 BenaFactory 与 AplicationConext 等 IOC 相关内容。
IoC(Inversion of control )即“控制反转”,它是一种设计思想而非一个技术实现。描述了Java 开发领域对象的创建以及管理的问题。通过Spring来管理对象的创建、配置和生命周期,这样相当于把控制权交给了Spring,不需要人工来管理对象之间复杂的依赖关系,这样做的好处就是解耦。
Spring 实现 IOC 的重要手段是依赖注入 DI 对象间的依赖的控制权从开发人员手中转移到了容器中,降低了开发成本.
在 Spring 里面,主要提供了 BeanFactory 和 ApplicationContext 两种 IOC 容器,通过他们来实现对 Bean 的管理。
IoC 最常见以及最合理的实现方式叫做依赖注入(Dependency Injection,简称 DI)。
依赖注入(DI,Dependency Injection)是 Spring 实现 IOC 的重要手段,依赖注入将对象间的依赖的控制权从开发人员手中转移到了容器中,降低了开发成本。
在Spring里面,主要提供了 BeanFactory 和 ApplicationContext 两种 IOC 容器,通过他们来实现对 Bean 的管理。
BenaFactory 可以说是 IOC 最顶层的接口,其定义了一个 IOC 容器的基本规范,可以说 BenaFactory 就是一个低配版的 IOC 容器,其定义了 IOC 容器最基本的功能
BeanFactory 使用控制反转对应用程序的配置和依赖性规范与实际的应用代码进行分离,BeanFactory 实例化后并不会自动实例化 Bean,只有当 Bean 被使用时才会对其进行实例化与依赖关系的装配。
public interface BeanFactory {
//对 FactoryBean 的转义定义,因为如果使用 bean 的名字检索 FactoryBean 得到的对象是工厂生成的对象,如果需要得到工厂本身,需要转义。
String FACTORY_BEAN_PREFIX = "&";
//根据 bean 的名字,获取在 IOC 容器中得到 bean 实例。
Object getBean(String name) throws BeansException;
//根据 bean 的名字和 Class 类型来得到 bean 实例,增加了类型安全验证机制。
<T> T getBean(String name, Class<T> requiredType) throws BeansException;
Object getBean(String name, Object... args) throws BeansException;
<T> T getBean(Class<T> requiredType) throws BeansException;
<T> T getBean(Class<T> requiredType, Object... args) throws BeansException;
<T> ObjectProvider<T> getBeanProvider(Class<T> requiredType);
<T> ObjectProvider<T> getBeanProvider(ResolvableType requiredType);
//提供对 bean 的检索,看看是否在 IOC 容器有这个名字的 bean。
boolean containsBean(String name);
//根据 bean 名字得到 bean 实例,并同时判断这个 bean 是不是单例。
boolean isSingleton(String name) throws NoSuchBeanDefinitionException;
//根据 bean 名字得到 bean 实例,并同时判断这个 bean 是不是原型。
boolean isPrototype(String name) throws NoSuchBeanDefinitionException;
//判断 bean 名字和 Type 判断 JavaBean 是否匹配指定的类型
boolean isTypeMatch(String name, ResolvableType typeToMatch) throws NoSuchBeanDefinitionException;
boolean isTypeMatch(String name, Class<?> typeToMatch) throws NoSuchBeanDefinitionException;
//得到 bean 实例的 Class 类型
@Nullable
Class<?> getType(String name) throws NoSuchBeanDefinitionException;
@Nullable
Class<?> getType(String name, boolean allowFactoryBeanInit) throws NoSuchBeanDefinitionException;
//得到 bean 的别名,如果根据别名检索,那么其原名也会被检索出来
String[] getAliases(String name);
}
可以看到,在 BeanFactory 里只对 IOC 容器的基本行为作了定义,根本不关心你的 Bean 是如何定义怎样加载的。而要知道 Bean 是如何定义怎样加载的,我们需要看具体的 IOC 容器实现,Spring 提供了许多 IOC 容器的 实现 。比如 GenericApplicationContext 与 ClasspathXmlApplicationContext 。
ApplicationContext 是 Spring 中的核心接口和容器,允许容器通过应用程序上下文环境创建、获取、管理 bean。在构建容器的时候,创建对象采用的策略是立即加载的方式,即只要一读取完配置文件就立即创建配置文件中配置的对象(BeanFactory采用的是延迟加载的方式,什么时候根据 id 获取对象了,什么时候才真正地创建对象)。
从以上类图我们可以看出 ApplicationContext 继承了 6 个接口,除了继承自 BeanFactory 的 HierarchicalBeanFactory 和 ListableBeanFactory 以外,还包括了:EnvironmentCapable、
ApplicationEventPublisher、ResourcePatternResolver 和 MessageSource。
### Spring 的三个主要实现类