Spring的BeanFactory与FactoryBean的区别
BeanFactory
BeanFactory
是 Spring
容器的顶级接⼝,给具体的IOC容器的实现提供了规范。
BeanFactory
只是个接口,并不是IOC容器的具体实现.
而 Spring
的本质是一个 Bean
工厂( BeanFactory )或者说 Bean
容器,它按照我们的要求,生产我们需要的各种各样的 Bean
,提供给我们使用。只是在生产 Bean
的过程中,需要解决 Bean
之间的依赖问题,才引入了依赖注入(DI)这种技术。也就是说依赖注入是 BeanFactory
生产 Bean
时为了解决 Bean
之间的依赖的一种技术而已。
BeanFactory 的实现
Spring
容器给出了很多种实现,如DefaultListableBeanFactory
、XmlBeanFactory
、ApplicationContext
等。而这些具体的容器都是实现了 BeanFactory
,再在其基础之上附加了其他的功能。
DefaultListableBeanFactory
BeanFactory
接口体系的默认实现类,实现以上接口的功能,提供 BeanDefinition
的存储 map
,Bean
对象对象的存储 map
。
其中 Bean
对象实例的存储 map
,定义在 FactoryBeanRegistrySupport
, FactoryBeanRegistrySupport
实现了 SingletonBeanRegistry
接口,而 DefaultListableBeanFactory
的基类 AbstractBeanFactory
,继承于 FactoryBeanRegistrySupport
。
StaticListableBeanFactory
用于存储给定的 bean 对象实例,不支持动态注册功能,是 ListableBeanFactory
接口的简单实现。
ApplicationContext
ApplicationContext
包含 BeanFactory
的所有功能,通常建议⽐ BeanFactory
优先。
ApplicationContext
以⼀种更向⾯向框架的⽅式⼯作以及对上下⽂进⾏分层和实现继承,ApplicationContext
包还提供了以下的功能:
-
MessageSource, 提供国际化的消息访问。
-
资源访问,如URL和⽂件
-
事件传播
-
载⼊多个(有继承关系)上下⽂ ,使得每⼀个上下⽂都专注于⼀个特定的层次,⽐如应⽤的web 层;
BeanFactory
提供的⽅法及其简单,仅提供了六种⽅法供客户调⽤:源码如下
String FACTORY_BEAN_PREFIX = "&";
//判断⼯⼚中是否包含给定名称的bean定义,若有则返回true
boolean containsBean(String beanName)
//返回给定名称注册的bean实例。根据bean的配置情况,如果是
singleton模式将返回⼀个共享实例,否则将返回⼀个新建的实例,如果没有找到指定bean,该⽅法
可能会抛出异常
Object getBean(String)
//返回以给定名称注册的bean实例,并转换为给定class类型
Class getType(String name) 返回给定名称的bean的Class,如果没有找到指定的bean实例,则排
除NoSuchBeanDefinitionException异常
Object getBean(String, Class)
//判断给定名称的bean定义是否为单例模式
boolean isSingleton(String)
//返回给定bean名称的所有别名
String[] getAliases(String name)
而 FactoryBean
是干什么用的呢?
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<T>
接⼝的 Bean
,根据该 Bean
的ID从BeanFactory
中获取的实际上是 FactoryBean
的getObject()
返回的对象,⽽不是 FactoryBean
本身,如果要获取 FactoryBean
对象,请在id前⾯加⼀个 &
符号来获取。
例如⾃⼰实现⼀个 FactoryBean
,功能:⽤来代理⼀个对象,对该对象的所有⽅法做⼀个拦截,在调⽤前后都输出⼀⾏LOG,模仿 ProxyFactoryBean
的功能。
FactoryBean
⼀个接⼝,当在IOC容器中的 Bean
实现了 FactoryBean
后,通过 getBean(StringBeanName)
获取到的 Bean
对象并不是 FactoryBean
的实现类对象,⽽是这个实现类中的 getObject()
⽅法返回的对象。要想获取 FactoryBean
的实现类,就要getBean(&BeanName)
,在 BeanName
之前加上&
.
在该接⼝中还定义了以下3个⽅法:
TgetObject():返回由FactoryBean创建的Bean实例,如果isSingleton()返回true,则该实例会放到Spring容器中单实例缓存池中;
booleanisSingleton():返回由FactoryBean创建的Bean实例的作⽤域是singleton还是
prototype;
ClassgetObjectType():返回FactoryBean创建的Bean类型。
总结
BeanFactory
是个 Factory
,也就是IOC容器或对象⼯⼚,FactoryBean
是个 Bean
。在 Spring
中,所有的 Bean
都是由 BeanFactory
(也就是IOC容器)来进⾏管理的。但对 FactoryBean
⽽⾔,这个 Bean
不是简单的 Bean
,⽽是⼀个能⽣产或者修饰对象⽣成的⼯⼚ Bean
,它的实现与设计模式中的⼯⼚模式和修饰器模式类似。