github地址:https://github.com/cheesezh/python_design_patterns

简单工厂模式 v.s. 工厂方法模式

以简单计算器为例,对比一下简单工厂模式和工厂方法模式的区别。

简单工厂模式

from abc import ABCMeta, abstractmethod

class Operation():
"""
抽象产品类(运算符类)
"""
__metaclass__ = ABCMeta def __init__(self):
self.result = None @abstractmethod
def get_result(self):
pass class AddOperation(Operation):
"""
具体产品类(加法运算符)
"""
def get_result(self, number_a, number_b):
self.result = number_a + number_b
return self.result class SubOperation(Operation):
"""
具体产品类(减法运算符)
"""
def get_result(self, number_a, number_b):
self.result = number_a - number_b
return self.result class MulOperation(Operation):
"""
具体产品类(乘法运算符)
"""
def get_result(self, number_a, number_b):
self.result = number_a * number_b
return self.result class DivOperation(Operation):
"""
具体产品类(除法运算符)
"""
def get_result(self, number_a, number_b):
if number_b == 0:
print("With operator '/', the second number can not be zero.")
return self.result
self.result = number_a / number_b
return self.result class OperationFactory():
"""
产品工厂类
"""
@classmethod
def create_operate(self, operator):
oper = None
if operator == "+":
oper = AddOperation()
elif operator == "-":
oper = SubOperation()
elif operator == "*":
oper = MulOperation()
elif operator == "/":
oper = DivOperation()
else:
print("Wrong operator.")
return oper number_a = int(input("input a number:"))
operator = str(input("input a operater(+ - * /):"))
number_b = int(input("input a number:")) oper = OperationFactory.create_operate(operator)
print(oper.get_result(number_a, number_b))
input a number:99
input a operater(+ - * /):/
input a number:9
11.0

工厂方法模式

from abc import ABCMeta, abstractmethod

class IFactory():
"""
通用工厂接口
"""
__metaclass__ = ABCMeta @abstractmethod
def create_operation(self):
pass class AddFactory(IFactory):
"""
实现工厂接口的加法工厂类
"""
def create_operation(self):
return AddOperation() class SubFactory(IFactory):
"""
实现工厂接口的剑法工厂类
"""
def create_operation(self):
return SubOperation() class MulFactory(IFactory):
"""
实现工厂接口的乘法工厂类
"""
def create_operation(self):
return MulOperation() class DivFactory(IFactory):
"""
实现工厂接口的除法工厂类
"""
def create_operation(self):
return DivOperation() def main():
number_a = int(input("input a number:"))
operator = str(input("input a operater(+ - * /):"))
number_b = int(input("input a number:")) if operator == "+":
oper_factory = AddFactory()
elif operator == "-":
oper_factory = SubFactory()
elif operator == "*":
oper_factory = MulFactory()
elif operator == "/":
oper_factory = DivFactory()
else:
print("Wrong operator.") oper = oper_factory.create_operation()
print(oper.get_result(number_a, number_b)) main()
input a number:99
input a operater(+ - * /):/
input a number:11
9.0

点评

工厂方法更复杂了?

如果需要增加其他运算,比如求M的N次方。

在简单工厂模式里,先增加一个求M的N次方的产品类,然后更改工厂类的if判断增加分支即可。

在工厂方法模式里,先增加一个求M的N次方的产品类,还要新增一个相关工厂类,最后还有修改客户端代码。

这就是简单工厂和工厂方法的区别所在。简单工厂模式的最大优点在于工厂类中包含了必要的逻辑判断,根据客户端的选择条件动态实例化相关类,对于客户端来说,去除了与具体产品的依赖。但是,如果要增加一个新的功能,比如求M的N次方,需要更改工厂类的if判断分支条件,修改原有的类?违背了开放-封闭原则,这可不是好方法。所以就需要工厂方法模式来处理。

工厂方法模式

工厂方法模式,定义一个用于创建对象的接口,让子类决定实例化哪一个类。工厂方法使一个类的实例化延迟到其子类[DP]。

相当于将简单工厂中的工厂类,变成了一个工厂抽象接口和多个具体生成对象的工厂,于是我们要增加求M的N次方的功能,就不需要更改工厂类,只需要增加此功能的运算类和相应的工厂类即可。这样整个工厂和产品体系其实都没有修改,而只是扩展,这就完全符合了开放-封闭原则。

但是,工厂方法模式是现实,客户端需要决定实例化哪一个工厂来实现运算类,选择判断的问题还是存在的,也就是说,工厂方法把简单工厂的内部逻辑判断移到了客户端代码来进行。要增加新功能,本来修改工厂类,现在修改客户端了。

题目

木叶学校组织学雷锋活动,让鸣人,小樱,佐助帮敬老院的老人扫地,洗衣,买米,如何实现?

class LeiFeng():

    def sweep(self):
print("扫地") def wash(self):
print("洗衣") def buy_rice(self):
print("买米") class Student(LeiFeng):
pass def main():
mingren = Student()
xiaoying = Student()
zuozhu = Student() mingren.sweep()
xiaoying.wash()
zuozhu.buy_rice()
main()
扫地
洗衣
买米

点评

  • 学生都会毕业,但是帮助老人是长期工作,所以每次不同的人帮助老人,都需要改客户端代码,而且老人不可能知道所有来帮忙的学生的名字;
  • 除了学生,社区志愿者也可以帮助老人

如何用简单工厂方法解决上述问题?

class Volunteer(LeiFeng):
pass class SimpleFactory(): @classmethod
def create_leifeng(self, leifeng_type):
self.leifeng = None
if leifeng_type == "学生":
self.leifeng = Student()
elif leifeng_type == "志愿者":
self.leifeng = Volunteer()
else:
print("ERROR LeiFeng Type")
return self.leifeng def main():
studentA = SimpleFactory.create_leifeng("学生")
studentA.buy_rice()
studentB = SimpleFactory.create_leifeng("学生")
studentB.wash()
studentC = SimpleFactory.create_leifeng("学生")
studentB.sweep() main()
买米
洗衣
扫地

点评

  • 好的地方,客户端的代码,如果要换志愿者,只需要换参数即可;
  • 坏的地方,在任何实例化的时候都需要写一句SimpleFactory.create_leifeng("学生"),这会导致大量重复,在修改为志愿者的时候非常麻烦,可以用工厂方法解决这个问题;
from abc import ABCMeta, abstractmethod

class ILeiFengFactory():
__metaclass__ = ABCMeta @abstractmethod
def create_leifeng(self):
pass class StudentFactory(ILeiFengFactory): def create_leifeng(self):
return Student() class VolunteerFactory(ILeiFengFactory): def create_leifeng(self):
return Volunteer() def main():
leifeng_factory = StudentFactory()
stu1 = leifeng_factory.create_leifeng()
stu2 = leifeng_factory.create_leifeng()
stu3 = leifeng_factory.create_leifeng() stu1.sweep()
stu2.wash()
stu3.buy_rice() main()
扫地
洗衣
买米

点评

  • 此时如果要将学生改成志愿者,只需要修改一行代码即可;
  • 工厂方法克服了简单工厂违背开放-封闭原则的缺点,又保持了封装对象创建过程的优点;

总结

简单工厂和工厂方法都是集中封装了对象的创建,使得要更换对象时,不需要做大的改动就可以实现,降低了客户程序和产品对象的耦合。

工厂方法是简单工厂模式的进一步抽象和推广,由于使用了多态性,工厂方法模式保持了简单工厂模式的优点,而且克服了它的缺点。但缺点是由于每加一个产品,就需要加一个产品工厂类,增加了额外开发量。

另外,工厂方法还是没有避免修改客户端的代码,可以利用反射解决避免分支判断的问题。

[Python设计模式] 第8章 学习雷锋好榜样——工厂方法模式的更多相关文章

  1. 设计模式C#实现(九)——工厂方法模式和简单工厂

    工厂方法模式:定义一个用于创建对象的接口,让子类决定实例化哪一个类.Factory Method使一个类的实例化延迟到其子类. 构成: 1.Product工厂方法创建的对象的接口 2.Concrete ...

  2. Java设计模式从精通到入门四 工厂方法模式

    工厂方法模式 属于23中设计模式中创建型类型. 核心思想:工厂提供创建对象的接口,由子类决定实例化哪一个子类. 来源 ​ 设计模式之禅中的例子,女娲造人,通过八卦炉来进行造人,没有烧熟的为白人,烧太熟 ...

  3. PHP学习之工厂方法模式

    <?php //工厂方法模式 interface Doing { function eat(); function sleep(); } class Cat implements Doing { ...

  4. [Python设计模式] 第1章 计算器——简单工厂模式

    github地址:https://github.com/cheesezh/python_design_patterns 写在前面的话 """ 读书的时候上过<设计模 ...

  5. [Python设计模式] 第28章 男人和女人——访问者模式

    github地址:https://github.com/cheesezh/python_design_patterns 题目 用程序模拟以下不同情况: 男人成功时,背后多半有一个伟大的女人: 女人成功 ...

  6. [Python设计模式] 第24章 加薪审批——职责链模式

    github地址:https://github.com/cheesezh/python_design_patterns 题目 用程序模拟以下情景 员工向经理发起加薪申请,经理无权决定,需要向总监汇报, ...

  7. [Python设计模式] 第20章 挨个买票——迭代器模式

    github地址:https://github.com/cheesezh/python_design_patterns 迭代器模式 迭代器模式,提供一种方法顺序访问一个聚合对象中各个元素,而又不暴露该 ...

  8. [Python设计模式] 第18章 游戏角色备份——备忘录模式

    github地址:https://github.com/cheesezh/python_design_patterns 题目 用代码模拟以下场景,一个游戏角色有生命力,攻击力,防御力等数据,在打Bos ...

  9. C#设计模式之二工厂方法模式(Factory Method Pattern)【创建型】

    一.引言 在上一篇文章中我们讲解了过渡的一种模式叫做[简单工厂],也有叫[静态工厂]的,通过对简单工厂模式得了解,我们也发现了它的缺点,就是随着需求的变化我们要不停地修改工厂里面的方法的代码,需求变化 ...

随机推荐

  1. poj2464扫描线好题,回头再做

    扫描线+区间更新 题解 /* st[i],ol[i]表示y坐标大于y[i]和小于y[i]的点 两颗线段树建立在y轴上,区间[l,r]ol线选在[l,r]时st的分数 每次查询完成后再更新一次 遍历每条 ...

  2. 【【C++ Primer 第15章】 虚析构函数

    学习资料 • C++中基类的析构函数为什么要用virtual虚析构函数 虚析构函数 1. 正文 直接的讲,C++中基类采用virtual虚析构函数是为了防止内存泄漏.具体地说,如果派生类中申请了内存空 ...

  3. POJ1151Atlantis 矩形面积并 扫描线 线段树

    欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 题目传送门 - POJ1151 题意概括 给出n个矩形,求他们的面积并. n<=100 题解 数据范围极小. 我们分3种 ...

  4. 微信小程序开发--第一个项目

    一:Hello World 1.AppId 2.打开开发者工具 3.显示效果 二:

  5. Java Swing提供的文件选择对话框 - JFileChooser

    JFileChooser()           构造一个指向用户默认目录的 JFileChooser. JFileChooser(File currentDirectory)           使 ...

  6. 洛谷 p1123 取数游戏【dfs】

    题目链接:https://www.luogu.org/problemnew/show/P1123 转载于:>>>>>> 题目描述 一个N×M的由非负整数构成的数字矩 ...

  7. 014.Docker Harbor+Keepalived+LVS+共享存储高可用架构

    一 多Harbor高可用介绍 共享后端存储是一种比较标准的方案,将多个Harbor实例共享同一个后端存储,任何一个实例持久化到存储的镜像,都可被其他实例中读取.通过前置LB组件,如Keepalived ...

  8. 使用SQL逆向生成PDM文件

    首先导出表结构,可以使用Navicat 或者DataGrip 生成SQL文件后使用PowerDesigner 指定数据库类型,选择SQL文件即可

  9. js数据结构之二叉树的详细实现方法

    数据结构中,二叉树的使用频率非常高,这得益于二叉树优秀的性能. 二叉树是非线性的数据结构,用以存储带有层级的数据,其用于查找的删除的性能非常高. 二叉树 数据结构的实现方法如下: function N ...

  10. 非常简单的方法实现ViewPager自动循环轮播

    非常简单的方法实现ViewPager自动循环轮播,见红色代码部分,其它的代码可以忽略不看. 简洁高效是我解决问题的首要出发点. package com.shuivy.happylendandreadb ...