Spring5源码(8)-BeanFactory和FactoryBean的区别

简介: Spring5源码(8)-BeanFactory和FactoryBean的区别


BeanFactory和FactoryBean是两个容易混淆的概念,很多人喜欢问两者之间的区别,其实两者之间并无内在联系。

  • BeanFactory接口:IoC容器的顶级接口,是IoC容器的最基础实现,也是访问Spring容器的根接口,负责对bean的创建,访问等工作。
  • FactoryBean接口:可以返回bean的实例的工厂bean,通过实现该接口可以对bean进行一些额外的操作,例如根据不同的配置类型返回不同类型的bean,简化xml配置等。在使用上也有些特殊,BeanFactory接口中有一个字符常量String FACTORY_BEAN_PREFIX = "&"; 当我们去获取BeanFactory类型的bean时,如果beanName不加&则获取到对应bean的实例;如果beanName加上&,则获取到BeanFactory本身的实例;FactoryBean接口对应Spring框架来说占有重要的地位,Spring本身就提供了70多个FactoryBean的实现。他们隐藏了实例化一些复杂的细节,给上层应用带来了便利。从Spring3.0开始,FactoryBean开始支持泛型。

下面来看FactoryBean的使用方式

1.简化xml配置,隐藏细节

使用Setter方法注入大量属性会造成配置文件臃肿,这时可以考虑使用FactoryBean来简化配置。

  • bean

package com.lyc.cn.v2.day01.factoryBean;
/**
* @author: LiYanChao
* @create: 2018-09-05 11:50
*/
public class Student {
   /** 姓名 */
   private String name;
   /** 年龄 */
   private int age;
   /** 班级名称 */
   private String className;
   // ...可能能会有更多的属性
   public Student() {
   }
   public Student(String name, int age, String className) {
       this.name = name;
       this.age = age;
       this.className = className;
   }
   public String getName() {
       return name;
   }
   public void setName(String name) {
       this.name = name;
   }
   public int getAge() {
       return age;
   }
   public void setAge(int age) {
       this.age = age;
   }
   public String getClassName() {
       return className;
   }
   public void setClassName(String className) {
       this.className = className;
   }
   @Override
   public String toString() {
       return "Student{" + "name='" + name + '\'' + ", age=" + age + ", className='" + className + '\'' + '}';
   }
}

package com.lyc.cn.v2.day01.factoryBean;
import org.springframework.beans.factory.FactoryBean;
/**
 * @author: LiYanChao
 * @create: 2018-09-05 11:49
 */
public class StudentFactoryBean implements FactoryBean<Student> {
    private String studentInfo;
    @Override
    public Student getObject() throws Exception {
        if (this.studentInfo == null) {
            throw new IllegalArgumentException("'studentInfo' is required");
        }
        // 分割属性
        String[] splitStudentInfo = studentInfo.split(",");
        if (null == splitStudentInfo || splitStudentInfo.length != 3) {
            throw new IllegalArgumentException("'studentInfo' config error");
        }
        // 创建Student并填充属性
        Student student = new Student();
        student.setName(splitStudentInfo[0]);
        student.setAge(Integer.valueOf(splitStudentInfo[1]));
        student.setClassName(splitStudentInfo[2]);
        return student;
    }
    @Override
    public Class<?> getObjectType() {
        return StudentFactoryBean.class;
    }
    @Override
    public boolean isSingleton() {
        return true;
    }
    public void setStudentInfo(String studentInfo) {
        this.studentInfo = studentInfo;
    }
}

Student是一个普通的类,StudentFactoryBean实现了FactoryBean接口,是一个FactoryBean。

  • xml

<bean id="student" class="com.lyc.cn.v2.day01.factoryBean.StudentFactoryBean" p:studentInfo="张三,25,三年二班"/>
  • 测试

//FactoryBean简化配置测试
System.out.println(xmlBeanFactory.getBean("student"));
System.out.println(xmlBeanFactory.getBean("&student"));
  • 结果

========测试方法开始=======
Student{name='张三', age=25, className='三年二班'}
com.lyc.cn.v2.day01.factoryBean.StudentFactoryBean@1ff8b8f
========测试方法结束=======
  • 分析
    xmlBeanFactory.getBean("student") 获取到的是StudentFactoryBean产生的实例,也就是Student类的实例;而xmlBeanFactory.getBean("&student")获取到的是StudentFactoryBean自己的实例。
2.返回不同Bean的实例
  • bean

package com.lyc.cn.v2.day01.factoryBean;
/**
 * 家具接口
 */
public interface Furniture {
    void sayHello();
}

package com.lyc.cn.v2.day01.factoryBean;
/**
 * 椅子
 * @author: LiYanChao
 * @create: 2018-09-30 17:35
 */
public class Chair implements Furniture{
    @Override
    public void sayHello() {
        System.out.println("我是一把椅子。");
    }
}

package com.lyc.cn.v2.day01.factoryBean;
/**
 * 桌子
 * @author: LiYanChao
 * @create: 2018-09-30 17:35
 */
public class Desk implements Furniture{
    @Override
    public void sayHello() {
        System.out.println("我是一个桌子。");
    }
}

package com.lyc.cn.v2.day01.factoryBean;
import org.springframework.beans.factory.FactoryBean;
/**
 * 家具工厂bean
 * @author: LiYanChao
 * @create: 2018-09-05 15:11
 */
public class FurnitureFactoryBean implements FactoryBean<Furniture> {
    private String furniture;
    @Override
    public Furniture getObject() throws Exception {
        if (null == furniture) {
            throw new IllegalArgumentException("'furniture' is required");
        }
        if ("chair".equals(furniture)) {
            return new Chair();
        } else if ("desk".equals(furniture)) {
            return new Desk();
        } else {
            throw new IllegalArgumentException("'furniture' type error");
        }
    }
    @Override
    public Class<?> getObjectType() {
        if (null == furniture) {
            throw new IllegalArgumentException("'furniture' is required");
        }
        if ("chair".equals(furniture)) {
            return Chair.class;
        } else if ("desk".equals(furniture)) {
            return Desk.class;
        } else {
            throw new IllegalArgumentException("'furniture' type error");
        }
    }
    @Override
    public boolean isSingleton() {
        return true;
    }
    public void setFurniture(String furniture) {
        this.furniture = furniture;
    }
}
  • xml

<bean id="furniture" class="com.lyc.cn.v2.day01.factoryBean.FurnitureFactoryBean" p:furniture="desk"/>
  • 测试

@Test
public void test12() {
    //FactoryBean简单工厂测试
    Furniture furniture = xmlBeanFactory.getBean("furniture", Furniture.class);
    furniture.sayHello();
}
  • 结果

========测试方法开始=======
我是一个桌子。
========测试方法结束=======
  • 说明
    新建了家具接接口和桌子、椅子实现类,通过xml文件配置,在FurnitureFactoryBean的getObject方法进行判断,并返回不同的家具类型实例。
3.FactoryBean源码

该接口的源码比较少,只声明了三个接口

public interface FactoryBean<T> {
    //返回此工厂管理的对象的实例(可能Singleton和Prototype)
    //如果此FactoryBean在调用时尚未完全初始化(例如,因为它涉及循环引用),则抛出相应的FactoryBeanNotInitializedException。
    //从Spring 2.0开始,允许FactoryBeans返回null 对象。工厂会将此视为正常使用价值; 在这种情况下,它不再抛出FactoryBeanNotInitializedException。
    //鼓励FactoryBean实现现在自己抛出FactoryBeanNotInitializedException,视情况而定。
    T getObject() throws Exception;
    //返回此FactoryBean创建的对象类型,默认返回null
    Class<?> getObjectType();
    //实例是否单例模式,默认返回true
    default boolean isSingleton() {
        return true;
    }
}





目录
相关文章
|
1月前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
76 2
|
1月前
|
数据采集 监控 前端开发
二级公立医院绩效考核系统源码,B/S架构,前后端分别基于Spring Boot和Avue框架
医院绩效管理系统通过与HIS系统的无缝对接,实现数据网络化采集、评价结果透明化管理及奖金分配自动化生成。系统涵盖科室和个人绩效考核、医疗质量考核、数据采集、绩效工资核算、收支核算、工作量统计、单项奖惩等功能,提升绩效评估的全面性、准确性和公正性。技术栈采用B/S架构,前后端分别基于Spring Boot和Avue框架。
|
21天前
|
存储 缓存 Java
Spring面试必问:手写Spring IoC 循环依赖底层源码剖析
在Spring框架中,IoC(Inversion of Control,控制反转)是一个核心概念,它允许容器管理对象的生命周期和依赖关系。然而,在实际应用中,我们可能会遇到对象间的循环依赖问题。本文将深入探讨Spring如何解决IoC中的循环依赖问题,并通过手写源码的方式,让你对其底层原理有一个全新的认识。
41 2
|
24天前
|
Java 关系型数据库 数据库
京东面试:聊聊Spring事务?Spring事务的10种失效场景?加入型传播和嵌套型传播有什么区别?
45岁老架构师尼恩分享了Spring事务的核心知识点,包括事务的两种管理方式(编程式和声明式)、@Transactional注解的五大属性(transactionManager、propagation、isolation、timeout、readOnly、rollbackFor)、事务的七种传播行为、事务隔离级别及其与数据库隔离级别的关系,以及Spring事务的10种失效场景。尼恩还强调了面试中如何给出高质量答案,推荐阅读《尼恩Java面试宝典PDF》以提升面试表现。更多技术资料可在公众号【技术自由圈】获取。
|
1月前
|
前端开发 Java 开发者
Spring生态学习路径与源码深度探讨
【11月更文挑战第13天】Spring框架作为Java企业级开发中的核心框架,其丰富的生态系统和强大的功能吸引了无数开发者的关注。学习Spring生态不仅仅是掌握Spring Framework本身,更需要深入理解其周边组件和工具,以及源码的底层实现逻辑。本文将从Spring生态的学习路径入手,详细探讨如何系统地学习Spring,并深入解析各个重点的底层实现逻辑。
66 9
|
2月前
|
Java Spring
Spring底层架构源码解析(三)
Spring底层架构源码解析(三)
166 5
|
2月前
|
XML Java 数据格式
Spring底层架构源码解析(二)
Spring底层架构源码解析(二)
|
2月前
|
Java 编译器 Spring
Spring AOP 和 AspectJ 的区别
Spring AOP和AspectJ AOP都是面向切面编程(AOP)的实现,但它们在实现方式、灵活性、依赖性、性能和使用场景等方面存在显著区别。‌
106 2
|
2月前
|
设计模式 JavaScript Java
Spring 事件监听机制源码
Spring 提供了事件发布订阅机制,广泛应用于项目中。本文介绍了如何通过自定义事件类、订阅类和发布类实现这一机制,并展示了如何监听 SpringBoot 启动过程中的多个事件(如 `ApplicationStartingEvent`、`ApplicationEnvironmentPreparedEvent` 等)。通过掌握这些事件,可以更好地理解 SpringBoot 的启动流程。示例代码展示了从事件发布到接收的完整过程。
|
2月前
|
XML Java 数据格式
手动开发-简单的Spring基于注解配置的程序--源码解析
手动开发-简单的Spring基于注解配置的程序--源码解析
53 0