概述

我们日常基本的使用中对BeanFactory、FactoryBean、factory-method、facotry-bean基本上不需要关系,但是本着对技术的好奇、敬畏、面试的角度,我们还是要了解一下它们到底时什么,有什么用。以后在工作中可能会使用到哦。

BeanFactory

这个最容易理解,bean工厂,bean的工厂,其实就是我们日常说是的spring容器,所有被spring管理的bean都在其中。

FactoryBean

为什么要使用FactoryBean

我们有多种方式可以向Spring容器中注册bean。
可以使用@Configuration注解结合@Bean注解向Spring容器中注册bean;
可以按照条件向Spring容器中注册bean;
可以使用@Import注解向容器中快速导入bean对象;
可以在@Import注解中使用ImportBeanDefinitionRegistrar向容器中注册bean。

一般情况下,Spring是通过反射机制利用bean的class属性指定实现类来实例化bean的。
但是如果示例化bean的过程比较复杂,按照传统的方式需要在标签中提供大量的配置信息,配置方式的灵活性是受限的,如果能够采用编码的方式定制bean实例化,使用起来比较方便。
下面讲解如何使用FactoryBean向Spring容器中注册bean。
生产bean的工厂,可以理解为就是一个生成bean的代理类。是Spring的特殊接口,工厂类接口(FactoryBean),为了解决复杂的实例化bean过程而被创造的,新版本增加了泛型类应对所有类型。

public interface FactoryBean<T> {
    T getObject() throws Exception;  //返回由FactoryBean创建的Bean实例
    Class<T> getObjectType();
    boolean isSingleton();
}

通过这个接口,我们可以看出使用的就是工厂方法模式,不同的子类实现FactoryBean接口,生成具体的bean。

创建复杂bean时简化配置

spring中如果bean在多参数的情况下需要在中提供大量的配置信息。
例如:

public   class  Car  {  
        private   int maxSpeed ;  
        private  String brand ;  
        private   double price ;  
  //getter/setter方法省略...
}
<bean id ="car" class="xxx.Car">
        <property name="maxSpeed" ref="100"/>
        <property name="brand" ref="Suzuki"/>
        <property name="price" ref="100000"/> 
</bean>

可以看到当属性越多时,标签就越复杂。
当使用了FactoryBean作为工厂类来生产Car对象时:

// 工厂类,实现 FactoryBean<T>接口
public class CarFactoryBean implements FactoryBean<Car>  { 
    private  String carInfo ; 
    // 重写getObject()方法,创建Car实例
    public Car getObject () throws  Exception  {  
        Car car =  new  Car () ;  
        // 将carInfo信息以逗号形式分割,分给各个属性,返回实例
        String []  infos =  carInfo.split ( "," ) ;  
        car.setBrand ( infos [ 0 ]) ;  
        car.setMaxSpeed ( Integer. valueOf ( infos [ 1 ])) ;  
        car.setPrice ( Double. valueOf ( infos [ 2 ])) ;  
        return car;  
    }  
    public  Class<Car> getObjectType () {  
        return Car.class ;  
    }  
    public boolean isSingleton () {  
        return false ;  
    }  
    public String getCarInfo () {  
        return this.carInfo ;  
    }  
  
    // 接受逗号分割符设置属性信息  
    public void setCarInfo ( String carInfo )   {  
        this.carInfo = carInfo;  
    }  
}
<bean id="car" class="com.baobaotao.factorybean.CarFactoryBean">
    <property name="carInfo" value="法拉利,400,2000000"/>
</bean>

这样就极大的简化了标签的代码量,在配置多个bean,多参数的场景下很实用。
当调用getBean(“car”)时,Spring通过反射机制发现CarFactoryBean实现了FactoryBean的接口,这时Spring容器就调用方法CarFactoryBean#getObject()方法返回Car实例。
如果希望获取CarFactoryBean的实例,则需要使用getBean(“&car”),这是spring的内部约定:在需要获取FactoryBean时需要在getBean(beanName)方法是在beanName前显示的加上&前缀。
getBean(“car”) -> Car实例;
getBean(“&car”) -> CarFactoryBean实例;

简化依赖

当有其他类想要依赖Car类时:

public class Person { 
    private Car car ;
    private void setCar(Car car){ this.car = car;  }  
}

// 这里用java config代替了<bean>标签,但这不是重点
@Configuration 
public class CarConfiguration { 
    @Bean 
    public CarFactoryBean carFactoryBean(){ 
      CarFactoryBean cfb = new CarFactoryBean();
      cfb.setBrand("Honda");
      cfb.setPiece(1984000);
      return cfb;
    }

    @Bean
    public Person aPerson(){ 
    	Person person = new Person();
      	// 注意这里的不同,这里直接调用carFactoryBean().getObject()方法返回Car实例化对象
   	 	person.setCar(carFactoryBean().getObject());
    	return person; 
    } 
}

小结

  • BeanFactory:是个Factory,是SpringIOC容器,是对象的工厂。
  • FactoryBean:是个Bean,一个能生产对象,修饰对象生产的工厂bean。

factory-method 与factory-bean

factory-method,factory-bean只有在spring中才有,springboot中不存在。
它们是标签的属性,用于创建bean对象,需要参数的话使用constructor-arg元素来指定它的参数。
factory-method可以用在静态方法创建类和非静态方法创建类两种,factory-bean用来指定创建bean的工厂bean,只能用在非静态方法创建bean的场景。
使用factory-method和factory-bean方式创建bean,我们可以自己定义一个类,然后在该类中定义一个方法,在该方法中完成spring bean的创建工作,其中factory-bean用于指定自己定义的类,factory-method用于指定创建bean的方法。

示例

  • 定义要创建的bean
public class Stu {
    public String stuId;
    public Stu(String stuId) {
        this.stuId = stuId;
    }
}
  • 定义工厂bean
public class StuFactory {
    static Map<String,Stu> stuMap = new HashMap<>();
    static{
        //初始化
        Stream.iterate(1,n->n+1).limit(5).map(String::valueOf).forEach(t-> stuMap.put(t, new Stu(t)));
    }
    
    //静态创建类,方法必须是静态(static修饰)
    public static Stu getStaticStu(String stuId){
        return stuMap.get(stuId);
    }

    //动态创建类
    public Stu getDynamicStu(String stuId){
        return new Stu(stuId);
    }
}

定义xml

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">

    <!--静态获取对象-->
    <bean id="staticStu" class="yudaosourcecode.factorymethod.StuFactory" factory-method="getStaticStu">
        <!--传入getStaticStu方法的参数-->
        <constructor-arg value="1"/>
    </bean>

    <!--生成对象的工厂-->
    <bean id="stuFactory" class="yudaosourcecode.factorymethod.StuFactory"/>
    <!--动态获取对象-->
    <bean id="dynamicStu" factory-bean="stuFactory" factory-method="getDynamicStu">
        <!--传入getDynamicStu方法的参数-->
        <constructor-arg value="11"/>
    </bean>
</beans>
  • 测试
@Test
public void springFactoryMethodConfig() {
    ApplicationContext ac = new ClassPathXmlApplicationContext("classpath:springFactoryMethodConfig.xml");
    //获取静态Bean
    Stu staticStu = ac.getBean("staticStu", Stu.class);
    System.out.println("staticStu: ");
    System.out.println(staticStu);
    //获取动态Bean
    Stu dynamicStu = ac.getBean("dynamicStu", Stu.class);
    System.out.println("dynamicStu: ");
    System.out.println(dynamicStu);
}
staticStu: 
yudaosourcecode.factorymethod.Stu@6ec8211c
dynamicStu: 
yudaosourcecode.factorymethod.Stu@7276c8cd

Process finished with exit code 0

参考

BeanFactory,FactoryBean;factory-bean,factory-method 区别