一.理解工厂模式

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

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

二.工厂模式的优点

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

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

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

三.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. Tomcat中配置Url直接访问本地其他磁盘

    在配置 Tomcat serserver.xml 中配置 <Context path="/image" docBase="E:\image" debug= ...

  2. php的explode()和implode()方法

    php 中,字符串与数组互转       拆分字符串 到数组 explode()    - -(其他语言中的 split) 将数组连接成字符串 implode() <?php $test = ' ...

  3. ansible 角色登陆

    用ansible 来管理远程的主机,最大的好处是方便,ansible不用在远程的主机上安装ansible的客户端,ansible只要能通过ssh连接上远程主机就 能对它进行管理.也就是说ansible ...

  4. java设计模式(详)

    http://www.runoob.com/design-pattern/design-pattern-tutorial.html

  5. Deep learning with Python 学习笔记(4)

    本节讲卷积神经网络的可视化 三种方法 可视化卷积神经网络的中间输出(中间激活) 有助于理解卷积神经网络连续的层如何对输入进行变换,也有助于初步了解卷积神经网络每个过滤器的含义 可视化卷积神经网络的过滤 ...

  6. js设计模式之发布/订阅模式模式

    一.前言 发布订阅模式,基于一个主题/事件通道,希望接收通知的对象(称为subscriber)通过自定义事件订阅主题,被激活事件的对象(称为publisher)通过发布主题事件的方式被通知. 就和用户 ...

  7. Re:从零开始的Spring Session(一)

    Session和Cookie这两个概念,在学习java web开发之初,大多数人就已经接触过了.最近在研究跨域单点登录的实现时,发现对于Session和Cookie的了解,并不是很深入,所以打算写两篇 ...

  8. 使用EntityManager批量保存数据

    @PersistenceContext EntityManager em; 从别的系统中定期同步某张表的数据,由于数据量较大,采用批量保存 JPA EntityManager的四个主要方法 ① pub ...

  9. MFC进程的创建销毁、线程的创建与交互

    进程的创建 STARTUPINFO si; //**成员DWORD dwFlags;表示结构体当中哪些成员有效.**STARTF_USESHOWWINDOW|STARTF_USEPOSITION PR ...

  10. SQL 获取时间格式

    2018-12-17 22:31:01  SQL    :  CONVERT(varchar(100), GETDATE(), 20) Oracle:   to_date(SYSDATE(),'yyy ...