Spring中的设计模式:工厂方法模式
导读
- 工厂方法模式是所有设计模式中比较常用的一种模式,但是真正能搞懂用好的少之又少,Spring底层大量的使用该设计模式来进行封装,以致开发者阅读源代码的时候晕头转向。
- 文章首发于微信公众号【码猿技术专栏】,原创不易,谢谢支持!!!
- 今天陈某分别从以下五个方面详细讲述一下工厂方法模式:
- 「从什么是工厂方法模式」
- 「通用框架实现」
- 「工厂方法模式的优点」
- 「工厂方法模式的升级」
- 「Spring底层如何使用工厂方法模式」
什么是工厂方法模式?
- 定义:定义一个用于创建对象的 接口,让子类决定实例化哪一个类。工厂方法使一个类的实例化延迟到其子类。
- 工厂方法模式通用类图如下:

- 在工厂方法模式中,抽象产品
Product负责定义产品的特性,实现对事物的抽象定义。 AbstractFactory是抽象工厂类,定义了一个抽象工厂方法。具体的如何创建产品由工厂实现类ConcreteFactory完成。
通用框架实现
- 工厂方法模式的变种有很多,陈某给出一个比较实用的通用框架。
抽象产品类:
public abstract class Product {
/**
* 公共逻辑方法
*/
public void method1(){}
/**
* 抽象方法:由子类实现,根据业务逻辑定义多个
*/
public abstract void method2();
}
具体产品类1,继承抽象产品类,如下:
public class Product1 extends Product {
/**
* 实现抽象产品类的抽象方法
*/
@Override
public void method2() {
}
}
具体产品类2,继承抽象产品类,如下:
public class Product2 extends Product {
/**
* 实现抽象产品类的抽象方法
*/
@Override
public void method2() {
}
}
抽象工厂类,必须定义一个工厂方法来自己实现具体的创建逻辑,如下:
public abstract class AbstractFactory {
/**
* 工厂方法,需要子类实现
* @param cls
* @param <T>
* @return
*/
public abstract <T extends Product> T create(Class<T> cls);
}
- 具体工厂类,使用了反射对具体产品的实例化,如下:
public class ConcreteFactory extends AbstractFactory {
@Override
public <T extends Product> T create(Class<T> cls) {
Product product=null;
try{
product= (Product) Class.forName(cls.getName()).newInstance();
}catch (Exception ex){
ex.printStackTrace();
}
return (T) product;
}
}
- 测试如下:
public static void main(String[] args) {
//创建具体工厂类
ConcreteFactory factory = new ConcreteFactory();
//调用工厂方法获取产品类1的实例
Product1 product1 = factory.create(Product1.class);
System.out.println(product1);
}
- 以上是简单的一个通用框架,读者可以根据自己的业务在其上拓展。
工厂方法模式的优点
- 良好的封装性,代码结构清晰,调用者不用关系具体的实现过程,只需要提供对应的产品类名称即可。
- 易扩展性,在增加产品类的情况下,只需要适当的修改工厂类逻辑或者重新拓展一个工厂类即可。
- 屏蔽了产品类,产品类的变化调用者不用关心。比如在使用
JDBC连接数据库时,只需要改动一个驱动的名称,数据库就会从Mysql切换到Oracle,极其灵活。
工厂方法模式的升级
- 在复杂的系统中,一个产品的初始化过程是及其复杂的,仅仅一个具体工厂实现可能有些吃力,此时最好的做法就是为每个产品实现一个工厂,达到一个工厂类只负责生产一个产品。
- 此时工厂方法模式的类图如下:

- 如上图,每个产品类都对应了一个工厂,一个工厂只负责生产一个产品,非常符合单一职责原则。
- 针对上述的升级过程,那么工厂方法中不需要传入抽象产品类了,因为一个工厂只负责一个产品的生产,此时的抽象工厂类如下:
public abstract class AbstractFactory {
/**
* 工厂方法,需要子类实现
*/
public abstract <T extends Product> T create();
}
Spring底层如何使用工厂方法模式?
- 工厂方法模式在Spring底层被广泛的使用,陈某今天举个最常用的例子就是
AbstractFactoryBean。 - 这个抽象工厂很熟悉了,这里不再讨论具体的作用。其实现了
FactoryBean接口,这个接口中getObject()方法返回真正的Bean实例。 AbstractFactoryBean中的getObject()方法如下:
public final T getObject() throws Exception {
//单例,从缓存中取,或者暴露一个早期实例解决循环引用
if (isSingleton()) {
return (this.initialized ? this.singletonInstance : getEarlySingletonInstance());
}
//多实例
else {
//调用createInstance
return createInstance();
}
}
//创建对象
protected abstract T createInstance() throws Exception;
- 从以上代码可以看出,创建对象的职责交给了
createInstance这个抽象方法,由其子类去定制自己的创建逻辑。 - 下图显示了继承了
AbstractFactoryBean的具体工厂类,如下:

- 其实与其说
AbstractFactoryBean是抽象工厂类,不如说FactoryBean是真正的抽象工厂类,前者只是对后者的一种增强,完成大部分的可复用的逻辑。比如常用的sqlSessionFactoryBean只是简单的实现了FactoryBean,并未继承AbstractFactoryBean,至于结论如何,具体看你从哪方面看了。
总结
- 工厂方法模式是一种常见的设计模式,但是真正能够用的高级,用的透彻还是有些难度的,开发者所能做的就是在此模式基础上思考如何优化自己的代码,达到易扩展、封装性强的效果了。
Spring中的设计模式:工厂方法模式的更多相关文章
- 深入浅出设计模式——工厂方法模式(Factory Method)
介绍在简单工厂模式中,我们提到,工厂方法模式是简单工厂模式的一个延伸,它属于Gof23中设计模式的创建型设计模式.它解决的仍然是软件设计中与创建对象有关的问题.它可以更好的处理客户的需求变化. 引入我 ...
- 4. 星际争霸之php设计模式--工厂方法模式
题记==============================================================================本php设计模式专辑来源于博客(jymo ...
- C++设计模式——工厂方法模式
本文版权归果冻说所有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文链接,否则保留追究法律责任的权利.» 本文链接:http://www.jellythink.com/arch ...
- 设计模式——工厂方法模式(Factory Method)
原文地址:http://www.cnblogs.com/Bobby0322/p/4179921.html 介绍 在简单工厂模式中,我们提到,工厂方法模式是简单工厂模式的一个延伸,它属于Gof23中设计 ...
- Java设计模式-工厂方法模式(Virtual Constructor/Polymorphic Factory)
工厂方法模式(Virtual Constructor/Polymorphic Factory) 工厂方法模式是类的创建模式,又叫做虚拟构造子模式(Virtual Constructor)或者多态性工厂 ...
- 设计模式--工厂方法模式(Factory method pattern)及应用
面向对象的好处: 通过封装,继承,多态把程序的耦合度降低. 用设计模式可以使程序更加灵活,容易修改,且易于复用. 1. 工厂方法模式 Define an interface for creating ...
- JAVA设计模式--工厂方法模式
工厂方法设计模式 抽象工厂角色: 这是工厂方法模式的核心,它与应用程序无关.是具体工厂角色必须实现的接口或者必须继承的父类.在java中它由抽象类或者接口来实现.具体工厂角色:它含有和具体业务逻辑有关 ...
- 我的Java设计模式-工厂方法模式
女朋友dodo闹脾气,气势汹汹的说"我要吃雪糕".笔者心里啊乐滋滋的,一支雪糕就能哄回来,不亦乐乎?! 但是,雪糕买回来了,她竟然说"不想吃雪糕了,突然想吃披萨" ...
- 设计模式 — 工厂方法模式(Factory Method)
在开发系统中,经常会碰到一个问题.现在需要实现的一些功能,但是这个功能模块以后一定是需要扩展的,那么现在开发中就不仅要实现现在的功能,还要考虑以后的扩展.那么为了系统的健壮,扩展就要遵循开闭原则(简单 ...
- C#设计模式--工厂方法模式
0.C#设计模式-简单工厂模式 设计模式: 工厂方法模式(Factory Method Pattern) 介绍:简单工厂模式是要在工厂类中通过数据来做个决策,在工厂类中的多个类中实例化出来其中一个要用 ...
随机推荐
- web前端 关于浏览器兼容的一些知识和问题解决
浏览器兼容 为什么产生浏览器兼容,浏览器兼容问题什么是浏览器兼容: 所谓的浏览器兼容性问题,是指因为不同的浏览器对同一段代码有不同的解析,造成页面显示效果不统一的情况. 浏览器兼容产生的原因: 因为不 ...
- 第四章、深入理解vue组件
4-1.使用组件的细节 a.使用is解决html出现bug 如下 table下面应该为tr,所以页面渲染的时候没有找到tr是有问题的,所以是有小bug,所以table中必须是tr b.改上面bug,t ...
- java算法--链表
虽然这个文章看着很多,但是大多是对于细节的讲解,如果想要快速了解,可以直接观看末尾代码.上面的代码内容都是来自于文章末尾的代码. 很重要的算法,也是比较简单的算法. 但是在java中,因为不存在c和c ...
- 【剑指Offer】简单部分每日五题 - Day 1
今天开始更新leetcode上<剑指Offer>的题解,先从简单难度开始.预计按下列顺序更新: 简单难度:每日5题 中等难度:每日3题 困难难度:每日1题 17 - 打印从1到最大的n位数 ...
- C语言程序设计(六) 循环控制结构
第六章 循环控制结构 循环结构:需要重复执行的操作 被重复执行的语句序列称为循环体 计数控制的循环 条件控制的循环 当型循环结构 直到型循环结构 for while do-while while(循环 ...
- 深度学习与人类语言处理-语音识别(part2)
上节回顾深度学习与人类语言处理-语音识别(part1),这节课我们将学习如何将seq2seq模型用在语音识别 LAS 那我们来看看LAS的Encoder,Attend,Decoder分别是什么 Lis ...
- 关于使用 Laravel 服务容器的优势介绍
如果说laravel框架的核心是什么,那么无疑是服务容器.理解服务容器的概念,对于我们使用laravel太重要了,应该说是否理解服务容器的概念是区分是否入门laravel的重要条件.因为整个框架正是在 ...
- centos7安装部署opentsdb2.4.0
写在前面 安装HBase 在HBase中创建表结构 安装配置并启动opentsdb 写在前面 最近因为项目需要在读opentsdb的一部分源码,后面会做个小结分享出来.本人是不大喜欢写这种安装部署的文 ...
- 计算广告中的CPM和eCPM
计算广告中的CPM和eCPM CPM和eCPM分别是什么? CPM(Cost per Mille ) : 千次展示付费.是针对广告主说的,你要花多少钱,购买一千次广告展示的机会.类似的还有CPC (C ...
- SpringMVC框架——转发与重定向
网上摘取一段大神总结的转发与重定向的区别,如下: 转发(服务端行为) 形式:request.getRequestDispatcher().forward(request,response) 转发在服务 ...