FactoryBean与BeanFactory讲解

746 阅读4分钟

开启掘金成长之旅!这是我参与「掘金日新计划 · 12 月更文挑战」的第30天,点击查看活动详情

一、FactoryBean:用来自定义Bean的创建逻辑

FactoryBean是接口,是为IOC容器中的Bean的实现提供了更加灵活的方式,FactoryBean在IOC容器的基础上,给Bean的实现加上了一个简单工厂模式和装饰模式。一般情况下实例化一个Bean对象:Spring通过反射机制利用bean的class属性指定实现类实例化Bean;在某些情况下,实例化Bean过程比较复杂,如果按照传统的方式,则需要在bean中提供大量的配置信息。配置方式的灵活性是受限的,这时采用编码的方式可能会得到一个简单的方案。

Spring为此提供了一个org.springframework.bean.factory.FactoryBean的工厂类接口,用户可以通过实现该接口, 然后在getObject()方法中灵活配置来定制实例化Bean的逻辑。FactoryBean接口对于Spring框架来说占用重要的地位,Spring自身就提供了70多个FactoryBean的实现——(xxxFactoryBean)。它们隐藏了实例化一些复杂Bean的细节,给上层应用带来了便利。从Spring3.0开始,FactoryBean开始支持泛型,即接口声明改为FactoryBean的形式。从BeanFactory及其实现类的getBean() 方法中获取到的Bean对象,实际上是FactoryBean的getObject()方法创建并返回的Bean对象,而不是FactoryBean本身

public interface FactoryBean<T> {
    String OBJECT_TYPE_ATTRIBUTE = "factoryBeanObjectType";

    @Nullable
    T getObject() throws Exception;

    @Nullable
    Class<?> getObjectType();

    default boolean isSingleton() {
        return true;
    }
}

方法一:getObject():返回Bean对象;

方法二,getObjectType() :返回的Bean对象的类型;

方法三,isSingleton() :是否是单例对象,true是单例,false是非单例;

实现比如SqlSessionFactoryBean:

public class SqlSessionFactoryBean implements FactoryBean<SqlSessionFactory>, InitializingBean, ApplicationListener<ApplicationEvent>

定义TestBean实现FactoryBean接口并实现getObject()和getObjectType()方法,其中在getObject()方法中实例化一个TestBean对象,并对message进行了一些特殊处理(区别于构造方法)。

@Component
public class TestBean implements FactoryBean {
 
    private String message;
 
    public TestBean() {
        this.message = "通过构造方法初始化实例";
    }
 
    @Override
    public Object getObject() throws Exception {
        TestBean testBean = new TestBean();
        testBean.message = "通过FactoryBean.getObject()创建实例";
        // 这里并不一定要返回TestBean自身的实例,可以是其他任何对象的实例
        return testBean;
    }
 
    @Override
    public Class<?> getObjectType() {
        return TestBean.class;
    }
 
    public String getMessage() {
        return message;
    }
 
}

TestBean测试类:

@RunWith(SpringRunner.class)
@SpringBootTest(classes = TestApplication.class)
public class FactoryBeanTest {
 
    @Autowired
    private ApplicationContext context;
 
    @Test
    public void test() {
        TestBean myBean1 = (TestBean) context.getBean("testBean");
        System.out.println("myBean1 = " + myBean1.getMessage());
        TestBean myBean2 = (TestBean) context.getBean("&testBean");
        System.out.println("myBean2 = " + myBean2.getMessage());
        System.out.println("myBean1.equals(myBean2) = " + myBean1.equals(myBean2));
 
    }
 
}

输出结果:

myBean1 = 通过FactoryBean.getObject()初始化实例
myBean2 = 通过构造方法初始化实例
myBean1.equals(myBean2) = false

结论:

可见当调用getBean("testBean")时,Spring通过反射机制发现TestBean实现了FactoryBean接口,这时Spring容器就调用接口方法TestBean#getObject()方法返回,这时返回的并不是FactoryBean本身,而是FactoryBean#getObject()方法内部所返回的对象TestBean,相当于FactoryBean#getObject()代理了getBean() 方法。如果希望获取TestBean的实例,则需要在使用getBean(beanName)方法时在beanName前显示的加上 "&"前缀,如getBean("&testBean")。

当通过getBean()方法获取到一个Bean时,返回的并不是xxxFactoryBean本身,而是其创建的Bean对象; 如果要获取xxxFactoryBean对象本身,请在参数前面加一个&符号来获取,即:getBean(&BeanName)

二、BeanFactory:负责生产和管理bean的一个工厂类顶层接口

BeanFactory是一个负责生产和管理bean的一个工厂类(接口)。在Spring中,BeanFactory是IOC容器的核心接口,它的职责包括:实例化、定位、配置应用程序中的对象及建立这些对象间的依赖。我们通过getBean()方法,传入参数——bean的名称或者类型,便可以从Spring容器中来获取bean。 

BeanFactory是用于访问Spring容器的根接口,是从Spring容器中获取Bean对象的基础接口,提供了IOC容器最基本的形式,给具体的IOC容器的实现提供了规范。BeanFactory只是个接口,并不是IOC容器的具体实现,Spring容器给出了很多种 BeanFactory的 扩展实现,如:DefaultListableBeanFactory、XmlBeanFactory、ApplicationContext等。

image.png

从类图中可以看出,我们更常用的ApplicationContext就是一个BeanFactory。ApplicationContext接口由BeanFactory接口派生而来,ApplicationContext包含BeanFactory的所有功能,还提供了以下更多的功能:

1、MessageSource, 提供国际化的消息访问 ;

2、资源访问,如URL和文件;

3、事件传播。

相比BeanFactory,通常建议优先使用ApplicationContext。

BeanFactory的一些是实现,除了ApplicationContext,还有很多,例如:

  • AbstractBeanFactory:抽象Bean工厂,绝大部分的实现类,都是继承于他
  • DefaultListableBeanFactory:Spring默认的工厂类
  • XmlBeanFactory:前期使用XML配置用的比较多的时候用的Bean工厂
  • AbstractXmlApplicationContext:抽象应用容器上下文对象
  • ClassPathXmlApplicationContext:XML解析上下文对象,用户创建Bean对象我们早期写Spring的时候用它

三、FactoryBean与BeanFactory区别

BeanFactory是个Factory,也就是IOC容器或对象工厂,FactoryBean是个Bean。 在Spring中,所有的Bean都是由BeanFactory(也就是IOC容器)来进行管理的。但对FactoryBean而言,这个Bean不是简单的Bean,而是一个能生产或者修饰对象生成的工厂Bean,它的实现与设计模式中的工厂模式和修饰器模式类似。