作者 夏秋, 

非常感谢这位作者的深入浅出的讲解.

 

前言

工厂模式,顾名思义就是我们可以通过一个指定的“工厂”获得需要的“产品”,在设计模式中主要用于抽象对象的创建过程,让用户可以指定自己想要的对象而不必关心对象的实例化过程。这样做的好处是用户只需通过固定的接口而不是直接去调用类的实例化方法来获得一个对象的实例,隐藏了实例创建过程的复杂度,解耦了生产实例和使用实例的代码,降低了维护的复杂性。本文会用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)

抽象工厂模式与工厂方法模式最大的区别在于,抽象工厂中的一个工厂对象可以负责多个不同产品对象的创建 ,这样比工厂方法模式更为简单、有效率。

结论

初学设计模式时会对三种工厂模式的实际应用比较困惑,其实三种模式各有优缺点,应用的场景也不尽相同:

  • 简单工厂模式适用于需创建的对象较少,不会造成工厂方法中的业务逻辑太过复杂的情况下,而且用户只关心那种类型的实例被创建,并不关心其初始化过程时,比如多种数据库(MySQL/MongoDB)的实例,多种格式文件的解析器(XML/JSON)等。
  • 工厂方法模式继承了简单工厂模式的优点又有所改进,其不再通过一个工厂类来负责所有产品的创建,而是将具体创建工作交给相应的子类去做,这使得工厂方法模式可以允许系统能够更高效的扩展。实际应用中可以用来实现系统的日志系统等,比如具体的程序运行日志,网络日志,数据库日志等都可以用具体的工厂类来创建。
  • 抽象工厂模式在工厂方法基础上扩展了工厂对多个产品创建的支持,更适合一些大型系统,比如系统中有多于一个的产品族,且这些产品族类的产品需实现同样的接口,像很多软件系统界面中不同主题下不同的按钮、文本框、字体等等。