设计模式第二谈:工厂方法模式
Posted 码农阿焦
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了设计模式第二谈:工厂方法模式相关的知识,希望对你有一定的参考价值。
这篇文章介绍23种常用设计模式中的工厂方法 - Factory Method模式及其Python实现。
1、什么是工厂方法模式
“四人帮”编写的《设计模式-可复用面相对象软件的基础》一书中给出定义是:定义一个用于创建对象的接口,让子类决定实例化哪一个类
,工厂方法
使一个类的实例化延迟到其子类。
是不是感到一头雾水,脑海中瞬间浮现出“黑人问号”?我们暂且先讲这个定义放到这,先看看下面的Python代码示例,然后回过头来理解这个定义。
2、Python代码实现考虑这样一个场景:
一家汽车生产公司分别接到宝马、奔驰的订单,开设两条汽车生产线,分别生产宝马、奔驰汽车。
以上场景如何通过面向对象的方式实现呢?了解过面向对象的童鞋们会想说,我们需要两个基类:一个是表示汽车生产工厂的工厂类,一个具有汽车通用性质的汽车类;然后定义汽车类的两个子类,分别用来实现宝马汽车和奔驰汽车的具体细节。
接下来,我们看一下如何使用Python实现上述思路。
Python实现汽车类
Car.py
from abc import ABC, abstractmethod
class Car(ABC):
"""汽车抽象类"""
@abstractmethod
def drive(self):
pass
@abstractmethod
def whistle(self):
pass
这里定义了一个抽象类Car
来表示汽车,并抽象了汽车的两个操作:驾驶和鸣笛。
Python实现宝马汽车类
CarBMW.py
from .Car import Car
class CarBMW(Car):
"""宝马汽车实例类"""
def drive(self):
print("驾驶宝马汽车")
def whistle(self):
print("宝马汽车鸣笛")
Python实现奔驰汽车类
BENZ.py
from .Car import Car
class CarBENZ(Car):
"""奔驰汽车实例类"""
def drive(self):
print("驾驶奔驰汽车")
def whistle(self):
print("奔驰汽车鸣笛")
这里使用类继承的方式定义类两个类:表示宝马汽车的BMW、表示奔驰汽车的BENZ。
Python实现工厂类
Factory.py
from abc import ABC, abstractmethod
class CarFactory(object):
"""抽象工厂类:汽车工厂"""
@abstractmethod
def create(self):
pass
这里定义了一个工厂类CarFactory并定义了一个方法create
.
Python实现宝马汽车工厂
from .CarBMW import CarBMW
from .CarFactory import CarFactory
class CarFactoryBMW():
"""宝马汽车工厂"""
def create(self):
return CarBMW()
Python实现奔驰汽车工厂
from .CarBENZ import CarBENZ
from .CarFactory import CarFactory
class CarFactoryBENZ():
"""奔驰汽车工厂"""
def create(self):
return CarBENZ()
main.py
from .CarFactoryBMW import CarFactoryBMW
from .CarFactoryBENZ import CarFactoryBENZ
if __name__ == "__main__":
# 实例化宝马汽车工厂
bmw_car_factory = CarFactoryBMW()
# 生产宝马汽车,并测试
bmw_car = bmw_car_factory.create()
bmw_car.whistle() # 宝马汽车鸣笛
bmw_car.drive() # 驾驶宝马汽车
# 实例化奔驰汽车工厂
benz_car_factory = CarFactoryBENZ()
# 生产奔驰汽车,并测试
benz_car = benz_car_factory.create()
benz_car.whistle() # 奔驰汽车鸣笛
benz_car.drive() # 驾驶奔驰汽车
这里模拟了创建一个工厂,并根据订单生产相应的汽车,最后对生产的汽车做测试。执行main.py
程序将会得到如下输出。
宝马汽车鸣笛
驾驶宝马汽车
奔驰汽车鸣笛
驾驶奔驰汽车
到目前为止,我们使用Python并基于面向对象的理念实现了对场景的模拟。
你有没有注意到,上述代码是一种对工厂模式的解释呢?让我们在回顾一下工厂方法的定义:定义一个用于创建对象的接口,让子类决定实例化哪一个类,使得一个类的实例化延迟到其子类
。
CarFactory
类解释了定义一个用于创建对象的接口
CarFactoryBMW
和CarFactoryBENZ
类解释了让子类决定实例化哪一个类
看完这些解释是不是有一种”啊,原来是这样“的感觉,如果有这样的感觉,可以不忙看下面的内容,找个旁边的小伙伴,像他普及一下什么是工厂方法模式;如果没有这样的感觉,可能还需要在意会一下。
3、工厂方法模式的结构
依据上述的Python代码实现,可以总结出:工厂方法模式
由抽象工厂、具体工厂、抽象产品和具体产品等4个要素组成,其中:
抽象工厂:声明工厂创建产品的抽象方法,该方法将会返回一个产品类型的对象
具体工厂:主要是实现抽象工厂定义的创建产品的抽象方法,将会完成具体产品的创建
抽象产品:声明产品的通用属性和功能,定义产品规范
具体产品:具体实现了产品的属性和功能,具体产品由具体工厂来创建
工厂方法
模式的UML类图如下图所示。
4、工厂方法模式的适用性
工厂方法
模式可应用于以下情况:
当一个类不知道它所必须创建的对象的类的时候
当一个类希望由它的子类来指定它所创建的对象的时候
当类创建对象的职责委托给多个帮助子类的某一个,并且你希望哪一个帮助子类是代理者这一信息局部化的时候
5、工厂方法模式的优缺点
优点:
用户只需要知道具体工厂的名称就可以得到产品,无需知道产品的具体创建过程
在系统增加新的产品时,只需要添加具体产品类和具体工厂类,无需对原工厂进行任何修改,满足开闭原则
缺点:
当系统增加新的产品时,就要添加具体的产品类和工厂类,增加了代码的复杂度
6、工厂方法模式的其他实现
假如提前了解到产品类型就只有固定的几个,可以使用一个具体的工厂就能完成任务时,可删除上述的抽象工厂类及其子类,仅使用一个工厂类就能完成,这时的工厂方法
模式也称为简单工厂
模式,这时的工厂类如下:
from .CarBMW import CarBMW
from .CarBENZ import CarBENZ
class CarFactory():
"""工厂类"""
def create(car_type):
if car_type == "BMW":
return CarBMW()
elif car_type == "BENZ":
return CarBENZ()
以上是关于设计模式第二谈:工厂方法模式的主要内容,如果未能解决你的问题,请参考以下文章