spring 中 spring 中的 FactoryBean 与BeanFactory 的区别
先说结论:
BeanFactory 是定义IOC容器的核心接口,XmlBeanFactory,ApplicationContext等都实现了该接口;
FactoryBean 是一个接口,实现该接口的bean,交由IOC管理时将提供一个除反射外的另一种构建bean的方式(getObject),用于生产需要复杂操作的bean.
接下来看看代码与示例:
首先来看看BeanFactory
IOC容器的核心接口,定义了最底层的规则.相关的知识点后续有空继续补充,今天主要看看FactoryBean;
FactoryBean
public interface FactoryBean<T> {
//获取bean的方法
T getObject() throws Exception;
//定义bean的类型
Class<?> getObjectType();
//设置是否为单例
boolean isSingleton();
}
FactoryBean 总共只有这三个方法,作用已经在代码上标明,接下来进入源码看看实现了FactoryBean的bean是如何被初始化的
//一个普通POJO
@Data
public class ADemo {
private String id;
private String name;
}
//一个实现了FactoryBean的bean,并将其交由IOC管理
@Component
public class FactoryBeanTest implements FactoryBean {
@Override
public Object getObject() throws Exception {
return new ADemo();
}
@Override
public Class<?> getObjectType() {
return ADemo.class;
}
@Override
public boolean isSingleton() {
return true;
}
}
//一个用于测试的main方法,打印用来测试的bean的值
@ComponentScan
@EnableAspectJAutoProxy
public class BeanMain {
public static void main(String[] args){
AnnotationConfigApplicationContext annotationConfigApplicationContext=new AnnotationConfigApplicationContext(SpringConfig.class);
System.out.println(annotationConfigApplicationContext.getBean("factoryBeanTest"));
System.out.println(annotationConfigApplicationContext.getBean("&factoryBeanTest"));
}
接下来看看源码中的实现
首先在DefaultListableBeanFactory (默认工厂实现)中的preInstantiateSingletons(预实例化方法)中判断了当前实例化的bean是不是一个FactoryBean(isFactoryBean(beanName))如果是则在beanName钱加了一个"&";
然后下一步,看关键代码到AbstractBeanFactory中的getObjectForBeanInstance(所有的bean实例化都会走到这个方法)中做了一个判断,判断逻辑为(没有实现FactoryBean或者bean的名称中带"&")此处为重点
接着看进入到FactoryBeanRegistrySupport从名称上可以知道他是一个FactoryBean注册的支持抽象类,该类提供了一系列的方法供FactoryBean使用,进入到getObjectFromFactoryBean方法,判断当前处理的对象是不是个单例并且是否存在于singletonObjects(单例对象缓存)中.接下来去获取对象(先从工厂缓存中拿,拿不到在去doGetObjectFromFactoryBean中)
然后是最终结果可以看到最终调用的是factory的getObject()方法,也就是实现FactoryBean接口中的getObject()
上述便是获取实现了FactoryBean接口的bean实例化的过程,那么如果没有实现会怎么办呢?
区别就在这,如果beanName前带了"&"或者没有实现FactoryBean,则通过反射实例化走的是普通bean实例化的逻辑
最后main方法打印出的结果如下,
ADemo(id=null, name=null)
demo.Bean.FactoryBeanTest@3e6f3f28