Python笔记之工厂模式


Posted in Python onNovember 20, 2019

工厂模式: “工厂”即表示一个负责创建其他类型的对象的类,通常情况下,一个工厂的对象会有一个或多个方法与之关联,这些方法用于创建不同类型的对象,工厂对象会根据客户端给方法传递的不同的参数或者客户端调用不同的方法返回不同的对象。

优点:对象的创建是可以根据需要单独创建的,但是使用工厂模式来创建对象有以下优点:

  • 松耦合,对象的创建是根据工厂类来进行的,与类本身的实现是独立开来的。
  • 对于客户端来说,不需要知道类的具体实现,只需要调用相应接口就可以得到需要的对象了,这其实是简化了客户端的相关实现。
  • 对于对象的修改只需要在工厂里面进行即可,包括添加新的对象,客户端只需要更改少量的代码,甚至可以不修改代码就可以达到要求。
  • 使用工厂接口,还可以重用已有的对象,不用去别处调用已有的对象或者重新创建一个对象。

工厂模式的3种实现形式(或者说3中变体):

  • 简单工厂模式:工厂类会提供一个接口,并根据客户端传入参数来创建相应的实例对象。(创建一个对象)
  • 工厂方法模式:需要定义一个基类,不同的子类则代表着不同类型的对象。相对于简单工厂模式而言,工厂方法模式具有更强的可定制性。(创建一个对象)
  • 抽象工厂模式:需要定义一个抽象工厂类,然后由不同的子类来创建不同系列的对象,一个系列即代表一组对象。(创建一组对象)

简单工厂模式示例:

from abc import ABCMeta, abstractmethod


class Flower(metaclass=ABCMeta):
 @abstractmethod
 def show_price(self):
  pass


class Rose(Flower):
 def show_price(self):
  print('Rose price: $99')


class Tulip(Flower):
 def show_price(self):
  print('Tulip price: $66')


class FlowerSimpleFactory:
 def get_flower(self, flower_type):
  return eval(flower_type)()


if __name__ == '__main__':
 flower_factory = FlowerSimpleFactory()
 rose = flower_factory.get_flower('Rose')
 tulip = flower_factory.get_flower('Tulip')
 rose.show_price()
 tulip.show_price()
Rose price: $99
Tulip price: $66

特点:接口根据客户端传入的参数即可返回对应的实例对象,甚至不用返回它的对象就可以进行对应的操作(比如示例中的工厂FlowerSimpleFactory中可以直接定义一个print_price方法来打印各种花的价格,而不是先返回对象,再由对象调用show_price方法来打印),即不会暴露对象的创建逻辑,客户端直接使用接口即可完成对象的创建,甚至创建对象之后的一些操作。

工厂方法模式示例:

from abc import ABCMeta, abstractmethod


class Flower(metaclass=ABCMeta):
 @abstractmethod
 def show_price(self):
  pass


class Rose(Flower):
 def show_price(self):
  print('Rose price: $99')


class Tulip(Flower):
 def show_price(self):
  print('Tulip price: $66')


class Lily(Flower):
 def show_price(self):
  print('Lily price: $33')


class FlowerShopFactory(metaclass=ABCMeta):
 def __init__(self):
  self.flowers = []
  self.stock_flowers()

 @abstractmethod
 def stock_flowers(self):
  pass

 def get_flowers(self):
  return self.flowers

 def add_flower(self, flower):
  self.flowers.append(flower)


class FlowerShop1(FlowerShopFactory):
 def stock_flowers(self):
  self.add_flower(Rose())
  self.add_flower(Tulip())


class FlowerShop2(FlowerShopFactory):
 def stock_flowers(self):
  self.add_flower(Rose())
  self.add_flower(Tulip())
  self.add_flower(Lily())


if __name__ == '__main__':
 flower_shop1 = FlowerShop1()
 for flower in flower_shop1.get_flowers():
  flower.show_price()

 flower_shop2 = FlowerShop2()
 for flower in flower_shop2.get_flowers():
  flower.show_price()
Rose price: $99
Tulip price: $66
Rose price: $99
Tulip price: $66
Lily price: $33

特点:工厂方法可以根据基类来定义不同的子类,如示例中的FlowerShop1和FlowerShop2,每个子类则代表“工厂”可以创建的一个“产品”。即对象的创建是通过继承的子类来完成的。

抽象工厂模式示例:

from abc import ABCMeta, abstractmethod


class MiniCar(metaclass=ABCMeta):
 @abstractmethod
 def show_size(self):
  pass


class SedanCar(metaclass=ABCMeta):
 @abstractmethod
 def show_price(self):
  pass


# 国产车
class DomesticMiniCar(MiniCar):
 def show_size(self):
  print('Domestic mini car size: 111')


class DomesticSedanCar(SedanCar):
 def show_price(self):
  print('Domestic sedan car price: 10W')


# 英国车
class EnglishMiniCar(MiniCar):
 def show_size(self):
  print('English mini car size: 222')
  

class EnglishSedanCar(SedanCar):
 def show_price(self):
  print('English sedan car price: 30w')


# 抽象工厂类
class CarFactory(metaclass=ABCMeta):
 @abstractmethod
 def create_mini_car(self):
  pass

 @abstractmethod
 def create_sedan_car(self):
  pass


# 国产车工厂类
class DomesticCarFactory(CarFactory):
 def create_mini_car(self):
  return DomesticMiniCar()
 
 def create_sedan_car(self):
  return DomesticSedanCar()


# 英国车
class EnglishCarFactory(CarFactory):
 def create_mini_car(self):
  return EnglishMiniCar()
 
 def create_sedan_car(self):
  return EnglishSedanCar()

特点:需要定义一个接口(如示例的抽象工厂类)来创建一系列的相关对象,如示例中的两个子类分别创建两个系列的对象(国产车和英国车),即对象的创建也是由子类来完成。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Python 相关文章推荐
Python中使用item()方法遍历字典的例子
Aug 26 Python
Python实现抓取网页生成Excel文件的方法示例
Aug 05 Python
python的pip安装以及使用教程
Sep 18 Python
Python2 Selenium元素定位的实现(8种)
Feb 25 Python
Python IDE Pycharm中的快捷键列表用法
Aug 08 Python
Python程序暂停的正常处理方法
Nov 07 Python
如何基于pythonnet调用halcon脚本
Jan 20 Python
Python中常用的高阶函数实例详解
Feb 21 Python
使用Keras实现简单线性回归模型操作
Jun 12 Python
Pycharm2020.1安装无法启动问题即设置中文插件的方法
Aug 07 Python
Python爬虫抓取论坛关键字过程解析
Oct 19 Python
pandas数据分组groupby()和统计函数agg()的使用
Mar 04 Python
Python常用模块logging——日志输出功能(示例代码)
Nov 20 #Python
将python2.7添加进64位系统的注册表方式
Nov 20 #Python
10个Python面试常问的问题(小结)
Nov 20 #Python
python使用pip安装SciPy、SymPy、matplotlib教程
Nov 20 #Python
Python笔记之facade模式
Nov 20 #Python
将python安装信息加入注册表的示例
Nov 20 #Python
如何使用Python脚本实现文件拷贝
Nov 20 #Python
You might like
解决文件名解压后乱码的问题 将文件名进行转码的代码
2012/01/10 PHP
PHP实现下载远程图片保存到本地的方法
2017/06/19 PHP
Javascript SHA-1:Secure Hash Algorithm
2006/12/20 Javascript
屏蔽网页右键复制和ctrl+c复制的js代码
2013/01/04 Javascript
原生Js实现简易烟花爆炸效果的方法
2015/03/20 Javascript
jquery实现弹出层登录和全屏层注册特效
2015/08/28 Javascript
Vue.js每天必学之数据双向绑定
2016/09/05 Javascript
详解.vue文件中监听input输入事件(oninput)
2017/09/19 Javascript
通过 JS 判断页面是否有滚动条的实现方法
2018/04/05 Javascript
JavaScript中创建原子的方法总结
2018/08/26 Javascript
使用validate.js实现表单数据提交前的验证方法
2018/09/04 Javascript
详解在网页上通过JS实现文本的语音朗读
2019/03/28 Javascript
JavaScript 如何计算文本的行数的实现
2020/09/14 Javascript
[04:45]DOTA2上海特级锦标赛主赛事第四日RECAP
2016/03/06 DOTA
[01:07:46]完美世界DOTA2联赛循环赛 Magma vs IO BO2第二场 11.01
2020/11/02 DOTA
python实现二分查找算法
2017/09/21 Python
对变量赋值的理解--Pyton中让两个值互换的实现方法
2017/11/29 Python
Python numpy 点数组去重的实例
2018/04/18 Python
python实现事件驱动
2018/11/21 Python
Falsk 与 Django 过滤器的使用与区别详解
2019/06/04 Python
python打开windows应用程序的实例
2019/06/28 Python
Python爬虫抓取技术的一些经验
2019/07/12 Python
Python自动采集微信联系人的实现示例
2020/02/28 Python
Python使用graphviz画流程图过程解析
2020/03/31 Python
Python轻量级web框架bottle使用方法解析
2020/06/13 Python
HTML5 用动画的表现形式装载图像
2016/03/08 HTML / CSS
基于HTML5 Canvas:字符串,路径,背景,图片的详解
2013/05/09 HTML / CSS
Pretty Little Thing爱尔兰:时尚女性服饰
2017/03/27 全球购物
世界上最大的曲棍球商店:Pro Hockey Life
2017/10/30 全球购物
餐饮主管岗位职责
2013/12/10 职场文书
项目施工员岗位职责
2014/03/09 职场文书
考试没考好检讨书(精选篇)
2014/11/16 职场文书
世界红十字日活动总结
2015/02/10 职场文书
2016年幼儿园万圣节活动总结
2016/04/05 职场文书
详解MySQL数据库千万级数据查询和存储
2021/05/18 MySQL
OpenCV图像变换之傅里叶变换的一些应用
2021/07/26 Python