Spring的核心是IOC和AOP,面试Java的基本上都会被问到它们是干什么的,可能你就简单的说了是控制反转与面向切面编程,那么具体细节就很可能被问懵。今天我们来讲一下IOC中围绕Bean的一个知识点,BeanFactory与FactoryBean。每天搞懂一个知识点,长期下来会让你受益匪浅。
一、区别
-
相同点:
BeanFactory是接口,提供了IOC容器最基本的形式,也就是SpringIOC所遵循的最底层和最基本的编程规范。
FactoryBean是接口,实现这个接口的类,在注册到BeanFactory中时,并不像其他类暴露的是自己,而是FactoryBean中的getObject()方法的返回对象。 -
不同点:
BeanFactory是个factory,也就是IOC容器或者对相关工厂,在Spring中,所有的Bean都是由BeanFactory,也就是IOC容器来进行管理的。
FactoryBean是个Bean,但是这个Bean不是一个简单的Bean,而是一个能生产或者修饰对象的工厂Bean,它的实现类似于设计模式中的工厂模式或者装饰者模式。
二、BeanFactory
比较熟悉的DefaultListableBeanFactory、XmlBeanFactory、ApplicationContext类都是BeanFactory的子类。
BeanFacotry是spring中比较原始的Factory。如XMLBeanFactory就是一种典型的BeanFactory。
原始的BeanFactory无法支持spring的许多插件,如AOP功能、Web应用等。ApplicationContext接口,它由BeanFactory接口派生而来,ApplicationContext包含BeanFactory的所有功能,通常建议比BeanFactory优先 。
ApplicationContext以一种更向面向框架的方式工作以及对上下文进行分层和实现继承,ApplicationContext包还提供了以下的功能:
- MessageSource, 提供国际化的消息访问;
- 资源访问,如URL和文件;
- 事件传播;
- 载入多个(有继承关系)上下文 ,使得每一个上下文都专注于一个特定的层次,比如应用的web层;
BeanFactory源码:
public interface BeanFactory {
String FACTORY_BEAN_PREFIX = "&";
Object getBean(String name) throws BeansException;
// 根据名称和类型获取,如果名称对应的bean不是requiredType类型的,就抛出错误
<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);
boolean containsBean(String name);
boolean isSingleton(String name) throws NoSuchBeanDefinitionException;
boolean isPrototype(String name) throws NoSuchBeanDefinitionException;
boolean isTypeMatch(String name, ResolvableType typeToMatch) throws NoSuchBeanDefinitionException;
boolean isTypeMatch(String name, Class<?> typeToMatch) throws NoSuchBeanDefinitionException;
@Nullable
Class<?> getType(String name) throws NoSuchBeanDefinitionException;
@Nullable
Class<?> getType(String name, boolean allowFactoryBeanInit) throws NoSuchBeanDefinitionException;
String[] getAliases(String name);
}
方法简介:
- boolean containsBean(String beanName)
判断工厂中是否包含给定名称的bean定义,若有则返回true - Object getBean(String)
返回给定名称注册的bean实例。根据bean的配置情况,如果是singleton模式将返回一个共享实例,否则将返回一个新建的实例,如果没有找到指定bean,该方法可能会抛出异常 - Object getBean(String, Class)
返回以给定名称给定类型注册的bean实例,如果bean不是给定Class的类型就抛出错误 - Class getType(String name)
返回给定名称的bean的Class,如果没有找到指定的bean实例,则排除NoSuchBeanDefinitionException异常 - boolean isSingleton(String)
判断给定名称的bean定义是否为单例模式 - String[] getAliases(String name)
返回给定bean名称的所有别名
三、FactoryBean
一般情况下,Spring通过反射机制利用<bean>的class属性指定实现类实例化Bean,在某些情况下,实例化Bean过程比较复杂,如果按照传统的方式,则需要在<bean>中提供大量的配置信息。配置方式的灵活性是受限的,这时采用编码的方式可能会得到一个简单的方案。Spring为此提供了一个org.springframework.bean.factory.FactoryBean的工厂类接口,用户可以通过实现该接口定制实例化Bean的逻辑。FactoryBean接口对于Spring框架来说占用重要的地位,Spring自身就提供了70多个FactoryBean的实现。它们隐藏了实例化一些复杂Bean的细节,给上层应用带来了便利。从Spring3.0开始,FactoryBean开始支持泛型,即接口声明改为FactoryBean<T>的形式
以Bean结尾,表示它是一个Bean,不同于普通Bean的是:它是实现了FactoryBean接口的Bean,根据该Bean的ID从BeanFactory中获取的实际上是FactoryBean的getObject()返回的对象,而不是FactoryBean本身,如果要获取FactoryBean对象,请在id前面加一个&符号来获取。