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

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

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

一、BeanFactory使用工厂方法

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

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

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

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对象。

既然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框架提高扩展能力。

这篇文章加深工厂方法的理解的同时,又进一步的深入了BeanFactory和FactoryBean<>的实现细节。

设计模式相关博客:

概述

一、设计模式概述

创建型设计模式

二、设计模式之工厂方法和抽象工厂

三、设计模式之单例和原型

四、设计模式之建造者模式

结构型设计模式

五、设计模式之代理模式

六、设计模式之适配器模式

七、设计模式之桥接模式

八、设计模式之组合模式

九、设计模式之装饰器模式

十、设计模式之外观模式

十一、设计模式之享元模式

行为型设计模式

十二、设计模式之责任链模式

十三、设计模式之命令模式

十四、设计模式之解释器模式

十五、设计模式之迭代器模式

十六、设计模式之中介者模式

十七、设计模式之备忘录模式

十八、设计模式之观察者模式

十九、设计模式之状态模式

二十、设计模式之策略模式

二十一、设计模式之模板方法模式

二十二、设计模式之访问者模式

工厂方法在Spring源码中的运用的更多相关文章

  1. Spring源码学习:第1步--在Spring源码中添加最简单的Demo代码

    为了最大程度地贴近Spring源码并进行学习,一种比较直接的做法是:直接在Spring源码中加入Demo代码,并进行调试. 参照以前使用Spring的经验,Spring最简单的使用方法是:一个实体类. ...

  2. Java设计模式:23种设计模式全面解析(超级详细)以及在源码中的应用

    从网络上找的设计模式, 很全面,只要把UML类图看懂了, 照着类图将代码实现是很容易的事情. 步骤: 先看懂类图, 然后将代码实现, 之后再看文字 http://c.biancheng.net/des ...

  3. CRUD搬砖两三年了,怎么阅读Spring源码?

    作者:小傅哥 博客:https://bugstack.cn 沉淀.分享.成长,让自己和他人都能有所收获! ‍连读同事写的代码都费劲,还读Spring? 咋的,Spring 很难读! 这个与我们码农朝夕 ...

  4. Spring源码阅读一

    引导: 众所周知,阅读spring源码最开始的就是去了解spring bean的生命周期:bean的生命周期是怎么样的呢,见图知意: 大致流程: 首先后通过BeanDefinitionReader读取 ...

  5. Spring 源码 (2)Spring IOC 容器 前戏准备工作

    Spring 最重要的方法refresh方法 根据上一篇文章 https://www.cnblogs.com/redwinter/p/16141285.html Spring Bean IOC 的创建 ...

  6. Spring 源码(8)Spring BeanPostProcessor的注册、国际化及事件发布机制

    上一篇文章https://www.cnblogs.com/redwinter/p/16198942.html介绍了Spring的注解的解析过程以及Spring Boot自动装配的原理,大概回顾下:Sp ...

  7. Spring 源码(9)Spring Bean的创建过程的前期准备

    回顾总结 到目前为止,Spring源码中AbstractApplicationContext#refresh方法的已经解读到第11个方法finishBeanFactoryInitialization, ...

  8. Spring 源码(10)Spring Bean 的创建过程(1)

    Spring Bean的创建刚开始进行了一些准备工作,比如转换服务的初始化,占位符解析器的初始化,BeanDefinition元数据的冻结等操作,都是为了在创建Bean的过程中保证Bean的正确的创建 ...

  9. Spring源码学习笔记12——总结篇,IOC,Bean的生命周期,三大扩展点

    Spring源码学习笔记12--总结篇,IOC,Bean的生命周期,三大扩展点 参考了Spring 官网文档 https://docs.spring.io/spring-framework/docs/ ...

随机推荐

  1. DZY Loves Math II

    简要题面 对于正整数 \(S, n\),求满足如下条件的素数数列 \((p_1,p_2,\cdots,p_k)\)(\(k\) 为任意正整数) 的个数: \(p_1\le p_2\le\cdots\l ...

  2. qbxt五一数学Day3

    目录 1. 组合数取模 1. \(n,m\le 200\),\(p\) 任意 2. \(n,m\le 10^6\),\(p\ge 10^9\) 素数 3. \(n,m\le 10^6\),\(p\le ...

  3. linux服务器监控脚本

    #!/bin/bash #获取cpu使用率 cpuUsage=`top -n 1 | awk -F '[ %]+' 'NR==3 {print $2}'` #获取磁盘使用率 data_name=&qu ...

  4. WPF 截图控件之画笔(八)「仿微信」

    前言 接着上周写的截图控件继续更新添加 画笔. 1.WPF实现截屏「仿微信」 2.WPF 实现截屏控件之移动(二)「仿微信」 3.WPF 截图控件之伸缩(三) 「仿微信」 4.WPF 截图控件之绘制方 ...

  5. CF383C Propagating tree (线段树,欧拉序)

    \(tag\)没开够\(WA\)了一发... 求出\(dfs\)序,然后按深度分类更新与查询. #include <iostream> #include <cstdio> #i ...

  6. Luogu1856 [USACO5.5]矩形周长Picture (线段树扫描线)

    对于横轴,加上与上一次扫描的差值:对于竖轴,加上高度差与区间内不相交线段\(*2\)的积: 难点在pushdown,注意维护覆盖关系.再就注意负数 #include <iostream> ...

  7. 深入理解Spring事件机制(一):广播器与监听器的初始化

    前言 Spring 从 3.x 开始支持事件机制.在 Spring 的事件机制中,我们可以令一个事件类继承 ApplicationEvent 类,然后将实现了 ApplicationListener ...

  8. [BJDCTF2020]Mark loves cat-1|源代码泄露|变量覆盖

    主要考察了:源代码泄露.变量覆盖 共展示了三种获取flag的方式 1.打开题目查看未发现有效信息,查看源代码信息,发现返回的dog信息,结果如下: 2.使用dirmap进行目录扫描,发现了.git/c ...

  9. 入门 Socket.io

    概念 Socket.io 是一个支持客户端和服务器之间的低延迟.双向和基于事件的通信的库,除了支持 JavaScript 以外,还支持 Java.Python.Golang. Socket.io 构建 ...

  10. 「学习笔记」单调队列优化dp

    目录 算法 例题 最大子段和 题意 思路 代码 修剪草坪 题意 思路 代码 瑰丽华尔兹 题意 思路 代码 股票交易 题意 思路 代码 算法 使用单调队列优化dp 废话 对与一些dp的转移方程,我们可以 ...