实例讲解Python设计模式编程之工厂方法模式的使用


Posted in Python onMarch 02, 2016

工厂方法模式是简单工厂模式的进一步抽象和推广,它不仅保持了简单工厂模式能够向客户隐藏类的实例化过程这一优点,而且还通过多态性克服了工厂类过于复杂且不易于扩展的缺点。在工厂方法模式中,处于核心地位的工厂类不再负责所有产品的创建,而是将具体的创建工作交由子类去完成。工厂方法模式中的核心工厂类经过功能抽象之后,成为了一个抽象的工厂角色,仅负责给出具体工厂子类必须实现的接口,而不涉及哪种产品类应当被实例化这一细节。工厂方法模式的一般性结构如下图所示,图中为了简化只给出了一个产品类和一个工厂类,但在实际系统中通常需要设计多个产品类和多个工厂类。

实例讲解Python设计模式编程之工厂方法模式的使用

工厂方法模式的实质是将对象的创建延迟到其子类实现,即由子类根据当前情况动态决定应该实例化哪一个产品类。从上图可以看出,工厂方法模式涉及到抽象工厂角色、具体工厂角色、抽象产品角色和具体产品角色四个参与者。

  • 抽象工厂(Creator)角色

    是工厂方法模式的核心,它负责定义创建抽象产品对象的工厂方法。抽象工厂不能被外界直接调用,但任何在模式中用于创建产品对象的工厂类都必须实现由它所定义的工厂方法。

  • 具体工厂(Concrete Creator)角色

    是工厂方法模式的对外接口,它负责实现创建具体产品对象的内部逻辑。具体工厂与应用密切相关,可以被外界直接调用,创建所需要的产品。

  • 抽象产品(Product)角色

    是工厂方法模式所创建的所有对象的父类,它负责描述所有具体产品共有的公共接口。

  • 具体产品(Concrete Product)角色

    是工厂方法模式的创建目标,所有创建的对象都是充当这一角色的某个具体类的实例。

抽象工厂角色负责声明工厂方法(factory method),用来"生产"抽象产品,以下是抽象工厂的示例性Python代码:

 creator.py
class Creator:
  """ 抽象工厂角色 """
  

# 创建抽象产品的工厂方法
 def factoryMethod(self):
 pass

具体工厂角色负责创建一个具体产品的实例,并将其返回给调用者。具体工厂是与具体产品相关的,实现时一般常用的做法是为每个具体产品定义一个具体工厂。以下是具体工厂的示例性Python代码:
concretecreator.py
class ConcreteCreator(Creator):
  """ 具体工厂角色 """
  

# 创建具体产品的工厂方法
 def factoryMethod(self):
 product = ConcreteProduct()
 return product

抽象产品角色的主要目的是为所有的具体产品提供一个共同的接口,通常只需给出相应的声明就可以了,而不用给出具体的实现。以下是抽象产品类的示例性Python代码:
product.py
class Product:
  """ 抽象产品角色 """

# 所有产品类的公共接口
 def interface(self):
 pass

具体产品角色充当最终的创建目标,一般来讲它是抽象产品类的子类,实现了抽象产品类中定义的所有工厂方法,实际应用时通常会具有比较复杂的业务逻辑。以下是具体产品类的示例性Python代码:
concreteproduct.py
class ConcreteProduct(Product):
  """ 具体产品角色 """
 

# 公共接口的实现
 def interface(self):
 print "Concrete Product Method"

 
在应用工厂方法模式时,通常还需要再引入一个客户端角色,由它负责创建具体的工厂对象,然后再调用工厂对象中的工厂方法来创建相应的产品对象。以下是客户端的示例性Python代码:
client.py
class Client:
  """ 客户端角色 """
  

def run(self):
 creator = ConcreteCreator()
 product = creator.factoryMethod()
 product.interface()
# 主函数
if (__name__ == "__main__"):
 client = Client()
 client.run()

在这个简单的示意性实现里,充当具体产品和具体工厂角色的类都只有一个,但在真正的实际应用中,通常遇到的都是同时会有多个具体产品类的情况,此时相应地需要提供多个具体工厂类,每个具体工厂都负责生产对应的具体产品。
工厂方法模式的活动序列如下图所示,客户端Client首先创建ConcreteCreator对象,然后调用ConcreteCreator对象的工厂方法factoryMethod(),由它负责"生产"出所需要的ConcreteProduct对象。

实例讲解Python设计模式编程之工厂方法模式的使用

下面我们来看一个具体案例:
如果你开一家Pizza店(PizzaStore抽象类)卖各种风味的Pizza(Pizza子类),那么你需要根据客户要求准备相应的Pizza(创建Pizza对象),然后烘烤、切片、包装;
最简单的做法就是在PizzaStore中根据客户要求(类型判断)创建相应的Pizza对象,然后调用Pizza自身(由Pizza抽象类实现)的烘烤、切片和包装方法;
但这样的代码缺乏弹性,因为你让一个抽象类去依赖具体的对象;我们可以创建一个工厂来生产Pizza,根据传入的不同类型值返回不同Pizza对象,即从PizzaStore中将创建对象的代码挪到工厂中。但这只是一个编程技巧,并不算模式。
在工厂方法模式中,我们在PizzaStore中定义一个抽象接口(create_pizza)作为抽象的工厂,而order_pizza是它的客户;将Pizza对象的创建放到PizzaStore子类去解决。
现有Cheese和Clam两款Pizza,以及NY和Chicago两家分店,每家店的同款Pizza的口味不同——为迎合当地口味做了改进,主要差别来自不同的原材料,因此我们实现四个Pizza类型(NYStyleCheesePizza、NYStyleClamPizza、ChicagoStyleCheesePizza和ChicagoStyleClamPizza),每种使用不同的原材料组合,根据客户所在城市和选择款式我们创建不同的对象;根据工厂方法,我们将对象创建的代码放到PizzaStore子类去实现。
代码:

#!/usr/bin/python 

class Pizza: 
 name = "" 
 dough = "" 
 sauce = "" 
 toppings = [] 
 
 def prepare(self): 
  print "Preparing %s" % self.name 
  print " dough: %s" % self.dough 
  print " sauce: %s" % self.sauce 
  print " add toppings:" 
  for n in self.toppings: 
   print "  %s" % n 
 
 def bake(self): 
  print "Bake for 25 minutes at 350." 
 
 def cut(self): 
  print "Cutting into diagonal slices." 
 
 def box(self): 
  print "Put into official box." 
 
 def get_name(self): 
  return self.name 
 
 
class PizzaStore: 
 def order_pizza(self, pizza_type): 
  self.pizza = self.create_pizza(pizza_type) 
  self.pizza.prepare() 
  self.pizza.bake() 
  self.pizza.cut() 
  self.pizza.box() 
  return self.pizza 
 
 def create_pizza(self, pizza_type): 
  pass 
 
 
class NYStyleCheesePizza(Pizza): 
 def __init__(self): 
  self.name = "NY Style Cheese Pizza" 
  self.dough = "NY Dough" 
  self.sauce = "NY Sauce" 
  self.toppings.append("NY toopping A") 
  self.toppings.append("NY toopping B") 
 
 
class ChicagoStyleCheesePizza(Pizza): 
 def __init__(self): 
  self.name = "Chicago Style Cheese Pizza" 
  self.dough = "Chicago Dough" 
  self.sauce = "Chicago Sauce" 
  sefl.toppings.append("Chicago toopping A") 
 
 def cut(self): 
  print "Cutting into square slices." 
 
 
class NYStyleClamPizza(Pizza): 
 def __init__(self): 
  self.name = "NY Style Clam Pizza" 
  self.dough = "NY Dough" 
  self.sauce = "NY Sauce" 
  self.toppings.append("NY toopping A") 
  self.toppings.append("NY toopping B") 
 
 
class ChicagoStyleClamPizza(Pizza): 
 def __init__(self): 
  self.name = "Chicago Style Clam Pizza" 
  self.dough = "Chicago Dough" 
  self.sauce = "Chicago Sauce" 
  self.toppings.append("Chicago toopping A") 
 
 def cut(self): 
  print "Cutting into square slices." 
 
 
class NYPizzaStore(PizzaStore): 
 def create_pizza(self, pizza_type): 
  if pizza_type == "cheese": 
   return NYStyleCheesePizza() 
  elif pizza_type == "clam": 
   return NYStyleClamPizza() 
  else: 
   return None 
 
 
class ChicagoPizzaStore(PizzaStore): 
 def create_pizza(self, pizza_type): 
  if pizza_type == "cheese": 
   return ChicagoStyleCheesePizza() 
  elif pizza_type == "clam": 
   return ChicagoStyleClamPizza() 
  else: 
   return None 
 
if __name__ == "__main__": 
 ny_store = NYPizzaStore() 
 chicago_store = ChicagoPizzaStore() 
 
 pizza = ny_store.order_pizza("cheese") 
 print "Mike ordered a %s." % pizza.get_name() 
 print 
 
 pizza = chicago_store.order_pizza("clam") 
 print "John ordered a %s." % pizza.get_name() 
 print

输出:

Preparing NY Style Cheese Pizza 
 dough: NY Dough 
 sauce: NY Sauce 
 add toppings: 
  NY toopping A 
  NY toopping B 
Bake for 25 minutes at 350. 
Cutting into diagonal slices. 
Put into official box. 
Mike ordered a NY Style Cheese Pizza. 
 
Preparing Chicago Style Clam Pizza 
 dough: Chicago Dough 
 sauce: Chicago Sauce 
 add toppings: 
  NY toopping A 
  NY toopping B 
  Chicago toopping A 
Bake for 25 minutes at 350. 
Cutting into square slices. 
Put into official box. 
John ordered a Chicago Style Clam Pizza.
Python 相关文章推荐
Python如何判断数独是否合法
Sep 08 Python
Python使用sftp实现上传和下载功能(实例代码)
Mar 14 Python
Python+OpenCV图片局部区域像素值处理改进版详解
Jan 23 Python
Python3.6中Twisted模块安装的问题与解决
Apr 15 Python
pytorch 改变tensor尺寸的实现
Jan 03 Python
OpenCV Python实现拼图小游戏
Mar 23 Python
基于python连接oracle导并出数据文件
Apr 28 Python
python轮询机制控制led实例
May 03 Python
Python实现像awk一样分割字符串
Sep 15 Python
python实现简单贪吃蛇游戏
Sep 29 Python
python海龟绘图之画国旗实例代码
Nov 11 Python
python实现简单聊天功能
Jul 07 Python
详解设计模式中的工厂方法模式在Python程序中的运用
Mar 02 #Python
探究python中open函数的使用
Mar 01 #Python
Saltstack快速入门简单汇总
Mar 01 #Python
通过Python使用saltstack生成服务器资产清单
Mar 01 #Python
Python解析json文件相关知识学习
Mar 01 #Python
使用简单工厂模式来进行Python的设计模式编程
Mar 01 #Python
Python文件处理
Feb 29 #Python
You might like
Mysql数据库操作类( 1127版,提供源码下载 )
2010/12/02 PHP
yii框架通过控制台命令创建定时任务示例
2014/04/30 PHP
php中addslashes函数与sql防注入
2014/11/17 PHP
php实现倒计时效果
2015/12/19 PHP
JavaScript中Object和Function的关系小结
2009/09/26 Javascript
jQuery操作input值的各种方法总结
2013/11/21 Javascript
深入理解JS中的变量及作用域、undefined与null
2014/03/04 Javascript
javascript实现按回车键切换焦点
2015/02/09 Javascript
JS选项卡动态替换banner图片路径的方法
2015/05/11 Javascript
动态加载jQuery的两种方法实例分析
2015/07/17 Javascript
自己动手制作基于jQuery的Web页面加载进度条插件
2016/06/03 Javascript
JS实现列表的响应式排版(推荐)
2016/09/01 Javascript
html判断当前页面是否在iframe中的实例
2016/11/30 Javascript
JS Input里添加小图标的两种方法
2017/11/11 Javascript
vue进行图片的预加载watch用法实例讲解
2018/02/07 Javascript
使用async await 封装 axios的方法
2018/07/09 Javascript
在小程序Canvas中使用measureText的方法示例
2018/10/19 Javascript
js删除对象/数组中null、undefined、空对象及空数组方法示例
2018/11/14 Javascript
vue使用codemirror的两种用法
2019/08/27 Javascript
JS正则表达式常见函数与用法小结
2020/04/13 Javascript
python中的随机函数小结
2018/01/27 Python
python实现键盘控制鼠标移动
2020/11/27 Python
Python列表list常用内建函数实例小结
2019/10/22 Python
python实现连续变量最优分箱详解--CART算法
2019/11/22 Python
python+opencv实现车牌定位功能(实例代码)
2019/12/24 Python
Python实现投影法分割图像示例(一)
2020/01/17 Python
python 遗传算法求函数极值的实现代码
2020/02/11 Python
Pytorch上下采样函数--interpolate用法
2020/07/07 Python
详解Python 最短匹配模式
2020/07/29 Python
Html5新标签解释及用法
2012/02/17 HTML / CSS
英国女性时尚鞋类的潮流制造者:Koi Footwear
2018/10/19 全球购物
新闻记者个人求职的自我评价
2013/11/28 职场文书
《吃水不忘挖井人》教学反思
2014/04/15 职场文书
学校学雷锋活动总结
2014/06/26 职场文书
2015年圣诞节活动总结
2015/03/24 职场文书
Elasticsearch 配置详解
2022/04/19 Java/Android