设计模式|工厂方法模式

一、工厂方法模式概念

工厂方法模式是类的创建模式,又叫做虚拟构造子(Cirtual Constructor)模式或者多态工厂(Polymorphic Factory)模式。 工厂方法模式的用意是定义一个创建产品对象的工厂接口,将实际创建工作推迟到子类中。

首先,在工厂方法模式中,核心的工厂类不再负责所有产品的创建,而是将具体创建的工作交给子类去做。这个核心类则摇身一变,成为了一个抽象工厂角色,仅负责给出具体工厂子类必须实现的接口,而不接触哪一个产品类应当被实例化这种细节。

这种进一步抽象化的结果,使这种工厂方法模式可以用来予许系统在不修改具体工厂角色的情况下引进新的产品,也就遵循了开闭原则。

二、工厂方法模式的结构

工厂方法模式的结构图如下: 从上图可以看出, 工厂方法模式涉及到抽象工厂角色,具体工厂角色,抽象产品角色以及具体产品角色等四个角色:

抽象工厂角色:担任这个角色的是工厂方法模式的核心,它是与应用程序无关的。任何在模式中创建对象的工厂类必须实现这个接口。

具体工厂角色:担任这个角色的是实现了抽象工厂接口的具体Java类,具体工厂角色含有与应用密切相关的逻辑,并且受到应用程序的调用以创建产品对象。

抽象产品角色:工厂方法模式所创建的对象的超类型,也就是产品对象的共同父类或共同拥有的接口。

具体产品角色:这个角色实现了抽象产品角色所申明的接口。工厂方法模式所创建的每一个对象都是某个具体产品角色的实例。

结合披萨系统,用白话文来说就是之前厨师(工厂类)负责所有的烤披萨任务,太累了。于是招了两个厨师分别负责烤 GreekPizza披萨和 CheesePizza披萨,之前的厨师升级为厨师长(抽象工厂类),负责教那两位厨师(具体工厂类)烤披萨,自己则不用亲自动手烤披萨了。

附上代码前先来看看完整的类图

三、代码示例

下面是抽象产品的角色Pizza的源代码:

public abstract class Pizza {
    public abstract void prepare();
    public abstract void bake();
    public abstract void cut();
    public abstract void box();
}

下面是具体产品角色CheesePizza的源代码:

public class CheesePizza extends Pizza{
    public void prepare(){
        System.out.println("准备CheesePizza~");
    }
    public void bake(){
        System.out.println("正在烤CheesePizza~");
    }
    public void cut(){
        System.out.println("正在切CheesePizza~");
    }
    public void box(){
        System.out.println("正在打包CheesePizza~");
    }
}

下面是具体产品角色GreekPizza的源代码:

public class GreekPizza  extends Pizza{
    public void prepare(){
        System.out.println("准备GreekPizza~");
    }
    public void bake(){
        System.out.println("正在烤GreekPizza~");
    }
    public void cut(){
        System.out.println("正在切GreekPizza~");
    }
    public void box(){
        System.out.println("正在打包GreekPizza~");
    }
}

下面是抽象工厂角色PizzaFactory的代码,这个角色是使用一个java接口实现,它声明了一个工厂方法,要求所有的具体工厂角色实现这个工厂方法:

public interface PizzaFactory {
    /**
     * 工厂方法
     * @return
     */
    public Pizza createPizza();
}

下面是具体工厂角色CheesePizzaFactory的代码,这个角色现实了抽象工厂角色PizzaFactory所声明的工厂方法:

public class CheesePizzaFactory implements PizzaFactory{
    @Override
    public Pizza createPizza() {
        return new CheesePizza();
    }
}

下面是具体工厂角色GreekPizzaFactory的代码,这个角色现实了抽象工厂角色PizzaFactory所声明的工厂方法:

public class GreekPizzaFactory  implements PizzaFactory{
    @Override
    public Pizza createPizza() {
        return new GreekPizza();
    }
}

下面是客户端角色的源代码:

public class OrderPizza {
    public static void main(String[] args){
        PizzaFactory factory=new CheesePizzaFactory();
        Pizza pizza=factory.createPizza();
        pizza.prepare();
        pizza.bake();
        pizza.cut();
        pizza.box();
        factory=new GreekPizzaFactory();
        pizza=factory.createPizza();
        pizza.prepare();
        pizza.bake();
        pizza.cut();
        pizza.box();
    }
}

结果演示:

准备CheesePizza~
正在烤CheesePizza~
正在切CheesePizza~
正在打包CheesePizza~
准备GreekPizza~
正在烤GreekPizza~
正在切GreekPizza~
正在打包GreekPizza~

这里使用工厂方法模式的注意点:

工厂方法创建对象:

工厂方法不一定每一次都返还一个新的对象,但是它所返还的对象一定是它自己创建的。

工厂方法返还的类型:

注意:工厂方法返还的应当是抽象类型,而不是具体类型,只有这样才能保证针对产品的多态性。当工厂方法模式发生上面的退化时,就不再是工厂方法模式了。

工厂等级结构:

工厂对象应当有一个抽象的超类型。换言之,应当有数个具体工厂类作为一个抽象超类型的具体子类存在于工厂等级结构中。如果等级结构中只有一个具体工程类的话,那么抽象工厂角色也可以省略,这时候,工厂方法模式就发生了退化,这一退化表现为针对工厂角色的多态性的丧失。

四、总结

工厂方法模式和简单工厂模式比较:

工厂方法模式跟简单工厂模式在结构上的不同是很明显的,工厂方法模式的核心是一个抽象工厂类,而简单工厂模式的核心在一个具体类。显而易见工厂方法模式这种结构更好扩展,权力下发,分布式比集中式更具优势。

如果系统需要加入一个新的产品,那么所需要的就是向系统中加入一个这个产品类以及它所对应的工厂类。没有必要修改客户端,也没有必要修改抽象工厂角色或者其他已有的具体工厂角色。对于增加新的产品类而言,这个系统完全支持开闭原则。

https://www.cnblogs.com/xiaoxi/p/7641358.html

END

好看、转发和辣条会提升颜值哦~

关注我

每天进步一点点

万水千山总是情,点个 “在看” 行不行

本文分享自微信公众号 - 技术从心(gh_d845efe513db)

原文出处及转载信息见文内详细说明,如有侵权,请联系 yunjia_community@tencent.com 删除。

原始发表时间:2019-03-30

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏腾讯高校合作

【犀牛鸟·学问】ACL2019最佳长论文: 桥接神经机器翻译的训练与推导(201907)— CCF-腾讯犀牛鸟基金线上学术报告

在2906篇投稿的激烈竞争中,本篇论文获得ACL2019最佳长论文奖项(Best Long Paper)。这篇文章研究应该如何桥接神经机器翻译训练和预测,是中...

14430
来自专栏AI科技时讯

七本书籍带你打下机器学习和数据科学的数学基础

大多数人学习数据科学的重心放在编程上面,然而,要真正精通数据科学的话是不能够忽视数据科学背后的数据基础。本篇文章,将分享给读者我喜欢的七本有关于数据科学基础的书...

8920
来自专栏APS-高级计划与排程

人工智能包括约束求解器吗?

以下是翻译Optaplanner创始人Geoffrey De Smet的一篇文章《Does A.I. include constraint solvers?》。...

9930
来自专栏腾讯技术工程官方号的专栏

腾讯数平团队 荣获第15届国际文档分析与识别竞赛七项冠军

? 导语:在刚刚结束的第15届国际文档分析与识别大会(澳大利亚悉尼)上,腾讯数据平台部(下称“数平”)团队获颁7项冠军证书,并受邀在会议上做技术分享。 9.2...

15440
来自专栏OSChina

设计模式-原型模式

7610
来自专栏有三AI

【AI-1000问】机器学习和模式识别是什么关系?

当今要是说起人工智能,聊起AI,每个人或多或少都能说出一点,从历史到未来,从图灵到冯诺依曼,从SVM到CNN等等,但是如果问你是否知道机器学习和模式识别有什么区...

7610
来自专栏腾讯高校合作

求实求新 | 2018年CCF-腾讯犀牛鸟基金成果分享(二)

在越发重视科技自主创新,新产业国际竞争逐渐激烈的时代,我们更加坚信,科研道路没有捷径可走,只有脚踏实地,一步一个脚印,不断积累方能实现创新。 7年来,犀牛鸟...

14460
来自专栏腾讯大数据的专栏

腾讯数平团队 荣获第15届国际文档分析与识别竞赛七项冠军

        9.20 - 9.25,作为全球OCR领域标杆性盛会,第15届国际文档分析与识别大会(ICDAR 2019)在澳大利亚悉尼召开,同时也揭晓了本...

11730
来自专栏深度学习与python

十幅插图带你了解CNN架构

LeNet-5是最简单的架构之一,它有2个卷积层和3个完全连接层。我们现在知道的平均汇集层被称为子采样层,它具有可训练的权重。该架构有大约60,000个参数。

9220

扫码关注云+社区

领取腾讯云代金券

年度创作总结 领取年终奖励