2. 工厂模式

工厂模式是创建型设计模式的一种。核心的思想是,通过传递给类或函数某种产品的信息来创建产品并返回。当我们想得到产品a对象,只需把产品a的名字传递给工厂函数就能得到产品a对象。而核心思想的背后是为了遵循著名的“开闭原则”:对扩展开放,对修改封闭。当添加新功能时,对已经有的函数代码不需要修改,只需要添加新功能相关的代码即可。

举一个最简单的例子:假如我们要写一个计算器,实现加减乘除四则运算,那么最简单的方法就是写一个函数,传入运算符,然后通过switch语句(python中没有switch语句,可以想办法用字典dict代替)。但问题也是显而易见的,如果我们要添加一个新的运算,比如开平方,那么就要修改原来的包含了四则运算的函数,显然,在这个过程中容易发生错误的修改,比如不小心改到了加法运算,这会让我们原先写好的功能有被修改错误的风险。所以,对应到这个问题上,工厂模式的核心就是要把这几个运算给完全独立出来,并且不影响运算主函数(即当添加新的运算时,我们不需要修改运算主函数)。怎么做呢?其实就是创建一个工厂类,由这个工厂类通过判断传入的运算符,去对应实例化各个运算函数(或者运算类)。这样,当添加了新的运算时,我们只需要添加新的运算函数(或者运算类),和修改工厂类(即添加新的switch判断,实例化新的运算函数或者运算类)就可以了。

2.1 简单工厂模式

简单工厂模式,是最简单的工厂方法模式,其由两部分组成,一是:工厂函数或工厂类,二是:产品类(可以是多个产品类A,B,C.........)。

下面我们通过一个简单的例子来看什么是简单工厂模式:

class A(object):
def __init__(self, product_type):
self.product_type = product_type def __str__(self):
return 'product %s' % self.product_type class B(object):
def __init__(self, product_type):
self.product_type = product_type def __str__(self):
return 'product %s' % self.product_type class Factory(object):
@staticmethod
def yield_product(product_type):
if product_type == 'A':
return A(product_type)
if product_type == 'B':
return B(product_type) if __name__ == '__main__':
a = Factory.yield_product('A')
b = Factory.yield_product('B')
print(a)
print(b)

结果如下

product A
product B

上面代码中,实现了两个产品类A,B,一个工厂类Factory,工厂类中实现了一个静态方法,其实,可以用一个工厂函数来代替这个工厂类。静态方法可以通过类名进行调用,而不需把类实例化。

静态方法@staticmethod的意思是,这个方法是一个普通方法,虽然属于类,但是不用访问类的其他成员。并且可以在不把类实例化的前提下,通过类名进行调用

2.2 工厂方法模式

简单工厂模式存在一个问题,当要增加产品时,需要修改工厂类或者工厂方法,在上面的例子中,我们需要修改工厂类的方法"yield_product"这违背了扩展开放,修改封闭的原则,因此是不可取的。工厂方法模式能够实现最小化地修改工厂类或工厂函数。

具体实现:对每一个产品类再进行一次封装,封装成只生产特定产品的工厂类。

import abc

class A(object):
def __init__(self, product_type):
self.product_type = product_type def __str__(self):
return 'product %s' % self.product_type class B(object):
def __init__(self, product_type):
self.product_type = product_type def __str__(self):
return 'product %s' % self.product_type class Abstract_factory(object):
__metaclass__ = abc.ABCMeta @abc.abstractmethod
def yield_product(self):
pass class Factory_a(Abstract_factory): def yield_product(self):
return A('A') class Factory_b(Abstract_factory): def yield_product(self):
return B('B') def factory(product_type):
if product_type == 'A':
return Factory_a()
if product_type == 'B':
return Factory_b() if __name__ == '__main__':
factory_a = factory('A')
a = factory_a.yield_product()
factory_b = factory('B')
b = factory_b.yield_product()
print(a)
print(b)

结果如下

product A
product B

上述代码,Factory_a,Factory_b是对类A,B的封装,一个工厂只负责创建一个产品遵从了单一职责原则,有利于代码的扩展和维护。

其中@abc.abstractmethod表示被该装饰器修饰的方法是一个抽象方法,没有实现,所以该基本不能被实例化。只有子类实现了该抽象方法才能被实例化。

参考链接:

1. python 设计模式(三) 工厂模式 https://blog.csdn.net/ruguowoshiyu/article/details/80657052

2. Python中的staticmethod与classmethod

3. python中@classmethod @staticmethod区别

4. Python2和Python3中@abstractmethod的用法 https://blog.csdn.net/xiemanR/article/details/72629164

设计模式之单例模式与工厂模式的Python实现(二)的更多相关文章

  1. 设计模式之单例模式与工厂模式的Python实现(一)

    1. 单例模式 单例模式(Singleton Pattern)是一种常用的软件设计模式,该模式的主要目的是确保某一个类只有一个实例存在.当你希望在整个系统中,某个类只能出现一个实例时,单例对象就能派上 ...

  2. 10月27日PHP加载类、设计模式(单例模式和工厂模式)、面向对象的六大原则

    加载类可以使用include.require.require_once三种中的任意一种,每个关键字都有两种方法,但是这种方法的缺点是需要加载多少个php文件,就要写多少个加载类的方法.一般也就需要加载 ...

  3. php部分---面向对象,设计模式(单例模式、工厂模式)、oop六大原则;

    一.单例模式目的:为了控制对象的数量(只能够有一个,相当于类的计划生育)做法1.将类的构造函数做成私有的2.在类里面做了一个公有的函数来造对象3.将该函数变为静态的4.在函数里面加控制 class R ...

  4. java:常用的两种设计模式(单例模式和工厂模式)

    一.单例模式:即一个类由始至终只有一个实例.有两种实现方式(1)定义一个类,它的构造方法是私有的,有一个私有的静态的该类的变量在初始化的时候就实例化,通过一个公有的静态的方法获取该对象.Java代码  ...

  5. PHP模式设计之单例模式、工厂模式、注册树模式、适配器模式、观察者模式

    php模式设计之单例模式 什么是单例模式? 单例模式是指在整个应用中只有一个实例对象的设计模式 为什么要用单例模式? php经常要链接数据库,如果在一个项目中频繁建立连接数据库,会造成服务器资源的很大 ...

  6. headfirst设计模式(5)—工厂模式体系分析及抽象工厂模式

    先编一个这么久不写的理由 上周我终于鼓起勇气翻开了headfirst设计模式这本书,看看自己下一个设计模式要写个啥,然后,我终于知道我为啥这么久都没写设计模式了,headfirst的这个抽象工厂模式, ...

  7. java设计模式-----1、简单工厂模式

    简单工厂模式是属于创建型模式,又叫做静态工厂方法(Static Factory Method)模式,但不属于23种GOF设计模式之一.简单工厂模式是由一个工厂对象决定创建出哪一种产品类的实例,简单来说 ...

  8. Java设计模式(1)工厂模式(Factory模式)

    工厂模式定义:提供创建对象的接口. 为何使用工厂模式 工厂模式是我们最常用的模式了,著名的Jive论坛,就大量使用了工厂模式,工厂模式在Java程序系统可以说是随处可见. 为什么工厂模式是如此常用?因 ...

  9. 工厂模式的python实现

    #1.什么是工厂模式 #2.工厂模式的分类 ''' 1. 简单工厂模式 2. 工厂方法模式 3. 抽象工厂方法模式 ''' #3.简单工厂模式的python实现 from abc import ABC ...

随机推荐

  1. Java Web入门经典扫描版

    全书共分4篇19章,其中,第一篇为“起步篇”,主要包括开启JavaWeb之门.不可不知的客户端应用技术.驾驭JavaWeb开发环境.JavaWeb开发必修课之JSP语法等内容:第二篇为“核心篇”,主要 ...

  2. NodeJS基础入门

    1.前端最主流的JavaScript运行环境 1>Node.js是一个基于Chrome V8引擎的JavaScript运行环境. 2>Node.js使用了一个事件驱动.非阻塞式I/O的模型 ...

  3. poj_1995_Raising Modulo Numbers

    Description People are different. Some secretly read magazines full of interesting girls' pictures, ...

  4. Linux-日期时间相关命令

    获取当前时间 date [root@VM_0_3_centos ~]# date Mon Mar 18 19:13:33 CST 2019 [root@VM_0_3_centos ~]# date相关 ...

  5. PHP Socket服务器搭建和测试

    1.socket服务器搭建思路 1) 目的:理解socket服务器工作机制 2) 思路:创建socket -> 把socket加入连接池 -> 处理接收信息 -> 握手动作 -> ...

  6. C# 输出结果有System.Byte[]

    byte[]类型直接输出或者调用ToString函数都会出现这个结果. 需要执行: byte[] a=new byte[10]; string text = "";for (int ...

  7. 读懂CCS链接命令文件(.cmd)

    链接器的核心工作就是符号表解析和重定位,链接命令文件则使得编程者可以给链接器提供必要的指导和辅助信息.多数时候,由于集成开发环境的存在,开发者无需了解链接命令文件的编写,使用默认配置即可.但若需要对计 ...

  8. 初见spark-03(高级算子)

    最近心情不是很好,但是需要调节自己,真的需要调节自己,还是要努力,这个世界有我喜欢的人,有我追求的人,也许真的是守的住寂寞,耐得住繁华吧. 不说别的了,今天我们来接受啊spark的高级算子的系列 1. ...

  9. java程序——凯撒加密

    古罗马皇帝凯撒在打仗时曾经使用过以下方法加密军事情报: 请编写一个程序,使用上述算法加密或解密用户输入的英文字串要求设计思想.程序流程图.源代码.结果截图.   设计思想:输入一个字符串,然后将其中每 ...

  10. WCF入门一[WCF概述]

    一.什么是WCF WCF是使用托管代码建立和运行面向服务(Service Oriented)应用程序的统一框架.它使得开发者能够建立一个跨平台的.安全.可信赖.事务性的解决方案,且能与已有系统兼容协作 ...