工厂方法在Spring源码中的运用

简介: 使用了Spring框架,我们就很少自己进行对象的创建了,而我们使用到的对象当然就是交给Spring的工厂模式来创建的了。

工厂方法在Spring源码中的运用


我们都知道Spring中IOC是使用的工厂模式,但是对于实现细节就一知半解了,今天这篇文章就带大家解读Spring中是如何使用工厂模式的。

上篇文章中我们懂了什么是工厂模式,这篇文章就带着学过的概念深入到Spring的内部看看究竟是怎么使用该模式的。

在Spring中使用工厂方法的是BeanFactory和FactoryBean<>接口。


一、BeanFactory使用工厂方法

使用了Spring框架,我们就很少自己进行对象的创建了,而我们使用到的对象当然就是交给Spring的工厂模式来创建的了。


其中BeanFactory是Spring容器的顶层接口,也是Bean工厂最上层的接口,其会有很多工厂实现例如ClassPathXmlApplicationContext、XmlWebApplicationContext其中最常见的就是DefaultListableBeanFactory,我们可以把BeanFactory看成是一种工厂方法模式。

image.png

在工厂方法模式中有四个角色:

1、抽象工厂(Abstract Factory):提供了创建产品的接口,调用者通过它访问具体工厂的工厂方法 createProduct() 来创建产品。

2、具体工厂(ConcreteFactory):主要是实现抽象工厂中的抽象方法,完成具体产品的创建。

3、抽象产品(Product):定义了产品的规范,描述了产品的主要特性和功能。

4、具体产品(ConcreteProduct):实现了抽象产品角色所定义的接口,由具体工厂来创建,它同具体工厂之间一一对应。


BeanFactory是角色中的抽象工厂,而getBean就相当于我们实例中的createProduct()方法,用于创建对象。

DefaultListableBeanFactory等实现的工厂类就是角色中的具体工厂。

产品就是在我们开发中加上注解的@Controller@Service@compent@Configuration的类,Spring会将他们当成产品使用工厂模式生成对象。

在我们createProduct方法中是直接创建的对象,Spring肯定不会这么的笨,我们接着看getBean方法的实现方法。

getBean方法中一系列链路最终调用到doGetBean方法用于创建对象,在doGetBean中创建对象使用的是设计模式中的代理模式


二、FactoryBean<>使用工厂方法

讲完BeanFactory,再介绍另外一个工厂模式的应用FactoryBean<>,想必大家也经常会比较这俩。


实际上,这两个接口都是用于创建对象,都可以看做是工厂方法模式的实现。

FactoryBean<>工厂通过getObject()方法来创建并返回对象,我们可以通过实现FactoryBean<>来定制化自己需要的Bean对象。

image.png

既然FactoryBean<>是留个开发者实现的,我们就手动实现一个FactoryBean<>。

@Component
public class MyFactoryBean<> implements FactoryBean<><Product>, BeanNameAware {
    private String name;
    @Override
    public Product getObject() throws Exception {
        return new Product();
    }
    @Override
    public Class<?> getObjectType() {
        return Product.class;
    }
    @Override
    public boolean isSingleton() {
        return false;
    }
    @Override
    public void setBeanName(String s) {
        this.name = name;
    }
}


接着我们分别使用两种方式getBean。

public static void main(String[] args) {
    //测试FactoryBean<>
    ApplicationContext ac =new AnnotationConfigApplicationContext(MyFactoryBean<>.class);
    System.out.println(ac.getBean("myFactoryBean<>"));
    System.out.println(ac.getBean("&myFactoryBean<>"));
}


只是在获取Bean时加了一个&会出现两种情况。

cn.sky1998.create.methodFactory.Product@52a86356
cn.sky1998.create.methodFactory.MyFactoryBean<>@5ce81285


doGetBean里面进行了判断是否是FactoryBean<>的实现类。

Spring对FactoryBean<>的实现机制是当你获取一个Bean时,如果获取的Bean的类型是FactoryBean<>,并且其name中并没有&则调用bean的getObject方法获取FactoryBean<>实现类中提供bean,否则就是直接返回普通的bean类型。

当我们在引入其他框架整合Spring时,便会有很多桥接整合包,例如mybatis-spring等,其中就会有FactoryBean<>的实现,例如SqlSessionFactoryBean<>、MapperFactoryBean<>等,将需要整合的定制化Bean通过工厂方法的模式,加入进Spring容器中。


当我们整合Mybatis时无论是多么的Bean都能很好的处理,mybatis-spring中的FactoryBean<>正是将Mybatis的核心启动类给调用出来。


三、BeanFactory和FactoryBean<>的异同

BeanFactory是Spring的一个大工厂,创建着Spring框架运行过程中所需要的Bean;

而FactoryBean<>是一个定制化工厂,其会存在于BeanFactory创建对象的过程中,当有需要时,会通过FactoryBean<>去自定制个性化的Bean,从而Spring框架提高扩展能力。


相关文章
|
1月前
|
缓存 Java 开发工具
Spring是如何解决循环依赖的?从底层源码入手,详细解读Spring框架的三级缓存
三级缓存是Spring框架里,一个经典的技术点,它很好地解决了循环依赖的问题,也是很多面试中会被问到的问题,本文从源码入手,详细剖析Spring三级缓存的来龙去脉。
134 24
Spring是如何解决循环依赖的?从底层源码入手,详细解读Spring框架的三级缓存
|
1月前
|
缓存 安全 Java
Spring框架中Bean是如何加载的?从底层源码入手,详细解读Bean的创建流程
从底层源码入手,通过代码示例,追踪AnnotationConfigApplicationContext加载配置类、启动Spring容器的整个流程,并对IOC、BeanDefinition、PostProcesser等相关概念进行解释
106 24
Spring框架中Bean是如何加载的?从底层源码入手,详细解读Bean的创建流程
|
1月前
|
XML 缓存 Java
手写Spring源码(简化版)
Spring包下的类、手写@ComponentScan注解、@Component注解、@Autowired注解、@Scope注解、手写BeanDefinition、BeanNameAware、InitializingBean、BeanPostProcessor 、手写AnnotationConfigApplicationContext
手写Spring源码(简化版)
|
4天前
|
JavaScript Java 关系型数据库
自主版权的Java诊所管理系统源码,采用Vue 2、Spring Boot等技术栈,支持二次开发
这是一个自主版权的Java诊所管理系统源码,支持二次开发。采用Vue 2、Spring Boot等技术栈,涵盖患者管理、医生管理、门诊管理、药店管理、药品管理、收费管理、医保管理、报表统计及病历电子化等功能模块。
|
10天前
|
Java Spring
Spring底层架构源码解析(三)
Spring底层架构源码解析(三)
|
10天前
|
XML Java 数据格式
Spring底层架构源码解析(二)
Spring底层架构源码解析(二)
|
21天前
|
Java Spring 容器
Spring IOC、AOP与事务管理底层原理及源码解析
【10月更文挑战第1天】Spring框架以其强大的控制反转(IOC)和面向切面编程(AOP)功能,成为Java企业级开发中的首选框架。本文将深入探讨Spring IOC和AOP的底层原理,并通过源码解析来揭示其实现机制。同时,我们还将探讨Spring事务管理的核心原理,并给出相应的源码示例。
74 9
|
16天前
|
设计模式 JavaScript Java
Spring 事件监听机制源码
Spring 提供了事件发布订阅机制,广泛应用于项目中。本文介绍了如何通过自定义事件类、订阅类和发布类实现这一机制,并展示了如何监听 SpringBoot 启动过程中的多个事件(如 `ApplicationStartingEvent`、`ApplicationEnvironmentPreparedEvent` 等)。通过掌握这些事件,可以更好地理解 SpringBoot 的启动流程。示例代码展示了从事件发布到接收的完整过程。
|
16天前
|
缓存 Java Spring
源码解读:Spring如何解决构造器注入的循环依赖?
本文详细探讨了Spring框架中的循环依赖问题,包括构造器注入和字段注入两种情况,并重点分析了构造器注入循环依赖的解决方案。文章通过具体示例展示了循环依赖的错误信息及常见场景,提出了三种解决方法:重构代码、使用字段依赖注入以及使用`@Lazy`注解。其中,`@Lazy`注解通过延迟初始化和动态代理机制有效解决了循环依赖问题。作者建议优先使用`@Lazy`注解,并提供了详细的源码解析和调试截图,帮助读者深入理解其实现机制。
17 1
|
18天前
|
存储 开发框架 Java
什么是Spring?什么是IOC?什么是DI?IOC和DI的关系? —— 零基础可无压力学习,带源码
文章详细介绍了Spring、IOC、DI的概念和关系,解释了控制反转(IOC)和依赖注入(DI)的原理,并提供了IOC的代码示例,阐述了Spring框架作为IOC容器的应用。
17 0
什么是Spring?什么是IOC?什么是DI?IOC和DI的关系? —— 零基础可无压力学习,带源码