一.理解工厂模式

在面向对象编程中,术语“工厂”表示一个负责创建替他类型对象的类。通常情况下,作为一个工厂的类有一个对象以及与它关联的多个方法。客户端使用某些参数调用此方法,之后,工厂会据此创建所需类型的对象,然后将它们返回给客户端。

简单理解:工厂模式依然是一种创建型设计模式,作为工厂,它所关心的是产品的产生,也就是对象的创建,我们利用工厂来创建对象,而不必我们亲自创建对象,我们无需去理解如何创建对象,只需要向工厂提出要求,让工厂去根据你的要求,给你生产你要的产品,给你相应的对象,这种模式便叫做工厂模式。

二.工厂模式的优点

  • 松耦合,对象的创建独立于类的实现

  • 客户端无需了解创建对象的类,只需知道需要传递的接口,方法和参数就能够创建所需要的对象

  • 很容易扩展工厂添加其他类型对象的创建,而使用者只需要改变参数就可以了

三.python实现工厂模式

1.简单工厂模式


from abc import ABCMeta,abstractmethod class Coke(metaclass=ABCMeta):
@abstractmethod
def drink(self):
pass class Coca(Coke):
def drink(self):
print('drink Coca-Cola') class Pepsi(Coke):
def drink(self):
print('drink Pepsi-Cola') class Fast_food_restaurant():
def make_coke(self ,name):
return eval(name)() KCD=Fast_food_restaurant()
coke=KCD.make_coke('Coca')
coke.drink()#drink Coca-Cola

ABCMeta是python的一个元类,用于在Python程序中创建抽象基类,抽象基类中声明的抽象方法,使用abstractmethod装饰器装饰。

eval(类名)返回的是一个class类型的对象

我们建立一个可乐的抽象类,百事可乐和可口可乐继承这个抽象类,我们又建立了快餐店类,也就是所说的工厂类,让它生产可乐。当用户需要可乐时,只需要告诉快餐店做一份什么品牌的可乐,告诉快餐店可乐的名字,然后快餐店使用make_coke方法做可乐,返回了你所需要的对象——一杯可口可乐,然后就可以快乐的喝可乐了。。

2.工厂方法模式

简单工厂模式已经帮我们做到我们需要某种对象时,可以不关心对象是怎么创建的,只需要向工厂类要对象即可,但是如果我们又多了一种对象,例如又出现了一个可乐品牌,嗯,我们叫它sfencs可乐吧,那么我们快餐店也得新添加这种可乐,也就是工厂类也得能够创建sfencs可乐对象了,但是这样就得在工厂类中加入新的逻辑判断来根据用户需求制造新添加的这个对象,显然是不恰当的,因为这样每当有一个新的类型的可乐增加的时候,我们都得修改工厂类的逻辑代码,使之能够判断出新的类型。这个问题使用工厂方法模式可以得到解决。

这里有一个小问题,如上面的简单工厂模式的代码,如果要新加sfencs可乐,似乎并不需要修改快餐店工厂类,只需要在sfencs可乐类定义之后,依然向make_coke函数传递类名即可,那这样岂不是不需要工厂方法模式了吗?

对于这个现象,其实是得益于eval()函数的功能,它能依据传入的字符串转换成相应的类,也就是eval函数就是工厂类的逻辑判断,如果不使用eval,那么逻辑判断可能就是多个if语句了,判断条件就是输入的字符串参数是否等于这个,是否等于那个。。。等。但是eval使用也有很大的局限性,它只能根据字符串来判断,我们不一定创建对象时都知道它的类名。除此之外,个人感觉eval这个函数有点厉害,越厉害就有可能造成不必要的麻烦,eval使用时还是得多注意。

接下来介绍工厂方法模式:


from abc import ABCMeta,abstractmethod class Coke(metaclass=ABCMeta):
@abstractmethod
def drink(self):
pass class Coca(Coke):
def drink(self):
print('drink Coca-Cola') class Pepsi(Coke):
def drink(self):
print('drink Pepsi-Cola') class Sfencs(Coke):
def drink(self):
print('drink Sfencs-Cola') class Fast_food_restaurant(metaclass=ABCMeta):
@abstractmethod
def make_coke(self):
pass class Coca_produce(Fast_food_restaurant):
def make_coke(self):
return Coca() class Pepsi_produce(Fast_food_restaurant):
def make_coke(self):
return Pepsi() class Sfencs_produce(Fast_food_restaurant):
def make_coke(self):
return Sfencs() KCD=Sfencs_produce()
coke=KCD.make_coke()
coke.drink()#drink Sfencs-Cola

工厂方法模式将原来的工厂类变为了抽象类,不同类型的可乐通过不同的子类生产,也就是工厂方法模式定义了一个创建对象的接口,但具体创建哪个类的对象由子类来决定,这种方式的逻辑判断相当于交给了客户端,也就是KCD=Sfencs_produce()来选择使用哪个子类,这样如果有新可乐产品出现的话,只需要再写一个子类继承工厂抽象类。

这里的类中,快餐店抽象类也叫做抽象工厂类,它的子类称为具体工厂类。可乐也一样,Coke为抽象产品类,它的子类为具体产品类。

3.抽象工厂模式

抽象工厂模式的主要目的是提供一个接口来创建一系列相关对象而无需指定具体的类。这个模式与与工厂方法模式的区别在于,它的一个方法子类,可以创建一系列的对象。

依然用可乐来举例,只喝普通的可乐还不足以让我们非常快乐,那么如果有一杯冰可乐,想必就能满足我们了。

于是我们的抽象产品类变为了两个,一个是冰可乐,一个是普通可乐,具体产品类有百事冰可乐、可口可乐冰可乐,普通百事,普通可口可乐。抽象工厂类有生产冰可乐和生产普通可乐的抽象方法,具体工厂类有百事工厂,可口可乐工厂。


from abc import ABCMeta,abstractmethod class Ice_coke(metaclass=ABCMeta):
@abstractmethod
def drink(self):
pass class Ordinary_coke(metaclass=ABCMeta):
@abstractmethod
def drink(self):
pass class Coca_ice(Ice_coke):
def drink(self):
print('drink Coca-ice-Cola') class Pepsi_ice(Ice_coke):
def drink(self):
print('drink Pepsi-ice-Cola') class Coca_ordinary(Ordinary_coke):
def drink(self):
print('drink Coca-ordinary-Cola') class Pepsi_ordinary(Ordinary_coke):
def drink(self):
print('drink Pepsi-ordinary-Cola') class Fast_food_restaurant(metaclass=ABCMeta):
@abstractmethod
def make_ice_coke(self):
pass @abstractmethod
def make_ordinary_coke(self):
pass class Coca_produce(Fast_food_restaurant):
def make_ice_coke(self):
return Coca_ice()
def make_ordinary_coke(self):
return Coca_ordinary() class Pepsi_produce(Fast_food_restaurant):
def make_ice_coke(self):
return Pepsi_ice()
def make_ordinary_coke(self):
return Pepsi_ordinary() KCD=Coca_produce()
coke=KCD.make_ice_coke()
coke.drink()#drink Coca-ice-Cola

这样再有其他类型的可乐或者其他品牌的可乐,只需要添加方法或者类就可以了。


参考《Python设计模式(第2版)》

python设计模式之工厂模式的更多相关文章

  1. python 设计模式之工厂模式 Factory Pattern (简单工厂模式,工厂方法模式,抽象工厂模式)

    十一回了趟老家,十一前工作一大堆忙成了狗,十一回来后又积累了一大堆又 忙成了狗,今天刚好抽了一点空开始写工厂方法模式 我看了<Head First 设计模式>P109--P133 这25页 ...

  2. Python: 设计模式 之 工厂模式例(2)(神奇的Python)

    #!/usr/bin/env python #coding=utf-8 # # 工厂模式第二例(神奇的Python) # 版权所有 2014 yao_yu (http://blog.csdn.net/ ...

  3. Python: 设计模式 之 工厂模式例(1)

    #!/usr/bin/env python #coding=utf-8 # # 工厂模式一例 # 版权所有 2014 yao_yu (http://blog.csdn.net/yao_yu_126) ...

  4. Python 设计模式--简单工厂模式

    简单工厂模式(Factory Pattern)是一种创建型的设计模式,像工厂一样根据要求生产对象实例. 特点:根据不同的条件,工厂实例化出合适的对象. <大话设计模式>中实例:四则运算计算 ...

  5. 浅谈Python设计模式 - 抽象工厂模式

    声明:本系列文章主要参考<精通Python设计模式>一书,并且参考一些资料,结合自己的一些看法来总结而来. 在上一篇我们对工厂模式中的普通工厂模式有了一定的了解,其实抽象工作就是 表示针对 ...

  6. 大话设计模式Python实现- 抽象工厂模式

    抽象工厂模式(Abstract Factory Pattern):提供一个创建一系列相关或相互依赖对象的接口,而无需指定它们的类 下面是一个抽象工厂的demo: #!/usr/bin/env pyth ...

  7. python设计模式之建造者模式

    python设计模式之建造者模式 ​ 建造者模式的适用范围:想要创建一个由多个部分组成的对象,而且它的构成需要一步接一步的完成.只有当各个部分都完成了,这个对象才完整.建造者模式表现为复杂对象的创建与 ...

  8. 设计模式——抽象工厂模式及java实现

    设计模式--抽象工厂模式及java实现 设计模式在大型软件工程中很重要,软件工程中采用了优秀的设计模式有利于代码维护,方便日后更改和添加功能. 设计模式有很多,而且也随着时间在不断增多,其中最著名的是 ...

  9. 5. 星际争霸之php设计模式--抽象工厂模式

    题记==============================================================================本php设计模式专辑来源于博客(jymo ...

随机推荐

  1. mysql 解决 timestamp 的2038问题

    当 timestamp 存储的时间大于 '2038-01-19 03:14:07' UTC,mysql就会报错,因为这是 mysql自身的问题,也就是说 timestamp是有上限的,超过了,自然会报 ...

  2. [Python] 震惊, 我居然用Python干这种事ꈍ .̮ ꈍ

    阅读本文只需花费你两分钟, 两分钟你买不了吃亏,你也买不了上当. 那么, 为何不静下心来看看呢? Python 海龟创意绘画, Turtle库创作精美图画 Author:Amd794     E-ma ...

  3. 数据结构与算法(周鹏-未出版)-第六章 树-6.5 Huffman 树

    6.5 Huffman 树 Huffman 树又称最优树,可以用来构造最优编码,用于信息传输.数据压缩等方面,是一类有着广泛应用的二叉树. 6.5.1 二叉编码树 在计算机系统中,符号数据在处理之前首 ...

  4. openerp7 时区问题解决--改成本地时区

    由于目前openerp的时区,读取的是UTC时间,而我国本地时间比UTC快8小时 问题就导致,写入数据库的时候时间相差8小时,以及Openerp日志输出时间格式也相差8小时和前端显示问题 1.更改op ...

  5. SpringMVC源码阅读入门

    1.导入 Spring Web MVC是基于Servlet API构建的原始Web框架,从一开始就包含在Spring框架中.正式的名称“Spring Web MVC”来自于它的源模块(spring-w ...

  6. .18-浅析webpack源码之compile流程-rules参数处理(1)

    Tips:写到这里,需要对当初的规则进行修改.在必要的地方,会在webpack.config.js中设置特殊的参数来跑源码,例如本例会使用module:{rules:[...]}来测试,基本上测试参数 ...

  7. [转]论magento1和magento2的速度性能优化问题

    本文转自:http://www.360magento.com/blog/magento-speed-up/ magento从2007年发展至今,也经历了十余年的磨练,如今也迎来了magento的换代产 ...

  8. html标签简介(常用)

    html常识 什么是HTML htyper text markup language  即超文本标记语言 超文本: 就是指页面内可以包含图片.链接,甚至音乐.程序等非文字元素. 标记语言: 标记(标签 ...

  9. 用Dreamweaver离线编写WordPress文章

    WordPress 自带的编辑器功能较弱,也不支持离线写文章.Windows Live Writer功能还行,不过早就停止更新了.如何能够找到一个功能强大的,可以离线编辑文章的WordPress文章编 ...

  10. mysql常用语句练习-基于ecshop2.7.3数据库(1)

    SELECT * FROM ecs_goods WHERE goods_id = 1;SELECT goods_id, goods_name FROM ecs_goods WHERE goods_id ...