我的编程空间,编程开发者的网络收藏夹
学习永远不晚

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

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

实例讲解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对象。

查看图片

下面我们来看一个具体案例:
如果你开一家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. 

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

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

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

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

工厂方法模式是简单工厂模式的进一步抽象和推广,它不仅保持了简单工厂模式能够向客户隐藏类的实例化过程这一优点,而且还通过多态性克服了工厂类过于复杂且不易于扩展的缺点。在工厂方法模式中,处于核心地位的工厂类不再负责所有产品的创建,而是将具体的创
2022-06-04

举例讲解Python设计模式编程的代理模式与抽象工厂模式

代理模式 Proxy模式是一种常用的设计模式,它主要用来通过一个对象(比如B)给一个对象(比如A) 提供'代理'的方式方式访问。比如一个对象不方便直接引用,代理就在这个对象和访问者之间做了中介 你先设想:一个对象提供rgb三种颜色值,我想获
2022-06-04

举例讲解Python设计模式编程中对抽象工厂模式的运用

抽象工厂模式:提供一个创建一系列相关或相互依赖对象的接口,而无需指定它们具体的类。 优点:易于交换“产品系列”,只要更改相应的工厂即可。 缺点:建立产品的时候很繁琐,需要增加和修改很多东西。 优化1:为了避免客户端有过多的逻辑判断,可以封装
2022-06-04

使用简单工厂模式来进行Python的设计模式编程

计模式的目的是让代码易维护、易扩展,不能为了模式而模式,因此一个简单的工具脚本是不需要用到任何模式的。 简单工厂模式又叫静态工厂方法模式,工厂模式家族中最简单的一种模式。这个模式的基本工作方式: 通过一个工厂来决定创建哪种具体的产品实例。
2022-06-04

java设计模式之工厂方法详解

一、概念 工厂方法模式是类的创建模式,又叫虚拟构造子模式(virtual constructor) 或者多态性工厂模式。二、模式动机 定义一个抽像的工厂,将产品的具体创建工作推迟到抽像工厂的具体子类中,有些
2023-05-31

使用Mixin设计模式进行Python编程的方法讲解

Mixin模式是一种在python里经常使用的模式,适当合理的应用能够达到复用代码,合理组织代码结构的目的。 Python的Mixin模式可以通过多继承的方式来实现, 举例来说,我们自定义一个简单的具有嵌套结构的数据容器:class Sim
2022-06-04

实例解析Python设计模式编程之桥接模式的运用

我们先来看一个例子:#encoding=utf-8 # #by panda #桥接模式 def printInfo(info): print unicode(info, 'utf-8').encode('gbk') #抽象类:手机品牌
2022-06-04

详解设计模式中的工厂方法模式在Python程序中的运用

工厂方法(Factory Method)模式又称为虚拟构造器(Virtual Constructor)模式或者多态工厂(Polymorphic Factory)模式,属于类的创建型模式。在工厂方法模式中,父类负责定义创建对象的公共接口,而子
2022-06-04

java设计模式学习之工厂方法模式如何实现

这篇文章主要为大家展示了“java设计模式学习之工厂方法模式如何实现”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“java设计模式学习之工厂方法模式如何实现”这篇文章吧。工厂方法模式(Facto
2023-05-31

Python设计模式编程中Adapter适配器模式的使用实例

将一个类的接口转换成客户希望的另外一个接口。使得原本由于接口不兼容而不能一起工作的那些类可以一起工作。 应用场景:希望复用一些现存的类,但是接口又与复用环境要求不一致。 模式特点:将一个类的接口转换成为客户希望的另外一个接口。 分类:类适配
2022-06-04

举例讲解Python设计模式编程中的访问者与观察者模式

访问者模式 我觉得Visitor模式是在补修改已有程序结构前提下,通过添加额外的访问者完成对代码功能的拓展 为什么这样用?当你的类层次较多,在某层结构中增加新的方法,要是在基类上面添加或者变更,可能破坏原来的设计, 有兼容问题,所以只在需要
2022-06-04

实例解析Java设计模式编程中的适配器模式使用

适配器模式的主要作用是在新接口和老接口之间进行适配,通过将一个类的接口转换成客户期望的另一个接口,让原本不兼容的接口可以合作无间,本文以实例解析Java设计模式编程中的适配器模式使用,需要的朋友可以参考下平时我们会经常碰到这样的情况,有了两
2023-05-31

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录