工厂模式,顾名思义就是我们可以通过一个指定的“工厂”获得需要的“产品”,在设计模式中主要用于抽象对象的创建过程,让用户可以指定自己想要的对象而不必关心对象的实例化过程。这样做的好处是用户只需通过固定的接口而不是直接去调用类的实例化方法来获得一个对象的实例,隐藏了实例创建过程的复杂度,解耦了生产实例和使用实例的代码,降低了维护的复杂性。
本文会用Python实现三种工厂模式的简单例子,所有代码都托管在Github上。
首先,我们先看一个简单工厂的例子:
1 #coding=utf-8 2 class Mercedes(object): 3 """梅赛德斯 4 """ 5 def __repr__(self): 6 return "Mercedes-Benz" 7 8 class BMW(object): 9 """宝马 10 """ 11 def __repr__(self): 12 return "BMW"
假设我们有两个“产品”分别是Mercedes和BMW的汽车,如果没有“工厂”来生产它们,我们就要在代码中自己进行实例化,如:
1 mercedes = Mercedes() 2 bmw = BMW()
但现实中,你可能会面对很多汽车产品,而且每个产品的构造参数还不一样,这样在创建实例时会遇到麻烦。这时就可以构造一个“简单工厂”把所有汽车实例化的过程封装在里面。
1 class SimpleCarFactory(object): 2 """简单工厂 3 """ 4 @staticmethod 5 def product_car(name): 6 if name == ‘mb‘: 7 return Mercedes() 8 elif name == ‘bmw‘: 9 return BMW()
有了SimpleCarFactory类后,就可以通过向固定的接口传入参数获得想要的对象实例,如下:
1 c1 = SimpleCarFactory.product_car(‘mb‘) 2 c2 = SimpleCarFactory.product_car(‘bmw‘)
虽然有了一个简单的工厂,但在实际使用工厂的过程中,我们会发现新问题:如果我们要新增一个“产品”,例如Audi的汽车,我们除了新增一个Audi类外还要修改SimpleCarFactory内的product_car方法。这样就违背了软件设计中的开闭原则[1],即在扩展新的类时,尽量不要修改原有代码。所以我们在简单工厂的基础上把SimpleCarFactory抽象成不同的工厂,每个工厂对应生成自己的产品,这就是工厂方法。
1 #coding=utf-8 2 import abc 3 4 class AbstractFactory(object): 5 """抽象工厂 6 """ 7 __metaclass__ = abc.ABCMeta 8 9 @abc.abstractmethod 10 def product_car(self): 11 pass 12 13 class MercedesFactory(AbstractFactory): 14 """梅赛德斯工厂 15 """ 16 def product_car(self): 17 return Mercedes() 18 19 class BMWFactory(AbstractFactory): 20 """宝马工厂 21 """ 22 def product_car(self): 23 return BMW()
我们把工厂抽象出来用abc模块实现了一个抽象的基类AbstractFactory,这样就可以通过特定的工厂来获得特定的产品实例了:
1 c1 = MercedesFactory().product_car() 2 c2 = BMWFactory().product_car()
每个工厂负责生产自己的产品也避免了我们在新增产品时需要修改工厂的代码,而只要增加相应的工厂即可。如新增一个Audi产品,只需新增一个Audi类和AudiFactory类。
工厂方法虽然解决了我们“修改代码”的问题,但如果我们要生产很多产品,就会发现我们同样需要写很多对应的工厂类。比如如果MercedesFactory和BMWFactory不仅生产小汽车,还要生产SUV,那我们用工厂方法就要再多构造两个生产SUV的工厂类。所以为了解决这个问题,我们就要再更进一步的抽象工厂类,让一个工厂可以生产同一类的多个产品,这就是抽象工厂。具体实现如下:
1 #coding=utf-8 2 import abc 3 4 # 两种小汽车 5 class Mercedes_C63(object): 6 """梅赛德斯 C63 7 """ 8 def __repr__(self): 9 return "Mercedes-Benz: C63" 10 11 class BMW_M3(object): 12 """宝马 M3 13 """ 14 def __repr__(self): 15 return "BMW: M3" 16 17 # 两种SUV 18 class Mercedes_G63(object): 19 """梅赛德斯 G63 20 """ 21 def __repr__(self): 22 return "Mercedes-Benz: G63" 23 24 class BMW_X5(object): 25 """宝马 X5 26 """ 27 def __repr__(self): 28 return "BMW: X5" 29 30 class AbstractFactory(object): 31 """抽象工厂 32 可以生产小汽车外,还可以生产SUV 33 """ 34 __metaclass__ = abc.ABCMeta 35 36 @abc.abstractmethod 37 def product_car(self): 38 pass 39 40 @abc.abstractmethod 41 def product_suv(self): 42 pass 43 44 class MercedesFactory(AbstractFactory): 45 """梅赛德斯工厂 46 """ 47 def product_car(self): 48 return Mercedes_C63() 49 50 def product_suv(self): 51 return Mercedes_G63() 52 53 class BMWFactory(AbstractFactory): 54 """宝马工厂 55 """ 56 def product_car(self): 57 return BMW_M3() 58 59 def product_suv(self): 60 return BMW_X5()
我们让基类AbstractFactory同时可以生产汽车和SUV,然后令MercedesFactory和BMWFactory继承AbstractFactory并重写product_car和product_suv方法即可。
1 c1 = MercedesFactory().product_car() 2 s1 = MercedesFactory().product_suv() 3 print(c1, s1) 4 s2 = BMWFactory().product_suv() 5 c2 = BMWFactory().product_car() 6 print(c2, s2)
抽象工厂模式与工厂方法模式最大的区别在于,抽象工厂中的一个工厂对象可以负责多个不同产品对象的创建 ,这样比工厂方法模式更为简单、有效率。
初学设计模式时会对三种工厂模式的实际应用比较困惑,其实三种模式各有优缺点,应用的场景也不尽相同:
原文:https://www.cnblogs.com/luke20181010/p/10574213.html