一、什么是工厂方法模式
简单工厂模式存在的问题: 类的创建依赖工厂类,也就是说,如果想要拓展程序,必须对工厂类进行修改,这违背了开闭原则。
工厂方法模式是一种常用的类创建型设计模式,此模式的核心精神是封装类中变化的部分,提取其中个性化善变的部分为独立类,通过依赖注入以达到解耦、复用和方便后期维护拓展的目的。
工厂方法模式的意义是定义一个创建产品对象的工厂接口,将实际创建工作推迟到子类当中。核心工厂类不再负责产品的创建,这样核心类成为一个抽象工厂角色,仅负责具体工厂子类必须实现的接口,这样进一步抽象化的好处是使得工厂方法模式可以使系统在不修改具体工厂角色的情况下引进新的产品。
工厂方法模式是简单工厂模式的衍生,解决了许多简单工厂模式的问题。实现开闭原则,实现了可扩展。其次更复杂的层次结构,可以应用于产品结果复杂的场合。
工厂方法模式之所以又被称为多态工厂模式,是因为所有的具体工厂类都具有同一抽象父类。
二、工厂方法模式的结构
角色 | 含义 |
---|---|
抽象工厂(Creator)角色 | 工厂方法模式的核心,与应用程序无关。任何在模式中创建的对象的工厂类必须实现这个接口 |
具体工厂(Concrete Creator)角色 | 实现抽象工厂接口的具体工厂类,包含与应用程序密切相关的逻辑,并且受到应用程序调用以创建产品对象 |
抽象产品(Product)角色 | 工厂方法模式所创建的对象的超类型,也就是产品对象的共同父类或共同拥有的接口 |
具体产品(Concrete Product)角色 | 实现了抽象产品角色所定义的接口。某具体产品有专门的具体工厂创建,它们之间往往一一对应 |
三、工厂方法模式的代码实现
python实现
from abc import ABCMeta, abstractmethod
# 抽象产品角色
class Payment(metaclass=ABCMeta):
@abstractmethod
def pay(self, money):
pass
# 具体产品角色一
class AliPayment(Payment):
def pay(self, money):
print(f'ali pay {money}.')
# 具体产品角色二
class WxPayment(Payment):
def pay(self, money):
print(f'wx pay {money}..')
# 抽象工厂角色
class PaymentFactory(metaclass=ABCMeta):
@abstractmethod
def create_payment(self):
pass
# 具体工厂角色一
class AliPaymentFactory(PaymentFactory):
def create_payment(self):
return AliPayment()
# 具体工厂角色二
class WxPaymentFactory(PaymentFactory):
def create_payment(self):
return WxPayment()
if __name__ == '__main__':
ali_factory = AliPaymentFactory()
ali = ali_factory.create_payment()
ali.pay(100)
wx_factory = WxPaymentFactory()
wx = wx_factory.create_payment()
wx.pay(10)
golang实现
package main
import "fmt"
type payment interface {
pay(money string)
}
type aliPayment struct {
payment
}
func (ali aliPayment) pay(money string) {
fmt.Println("this is ali pay: " + money)
}
type wxPayment struct {
payment
}
func (wx wxPayment) pay(money string) {
fmt.Println("this is wx pay: " + money)
}
type factory interface {
create() payment
}
type aliFactory struct {
factory
}
func (ali aliFactory) create() payment {
return aliPayment{}
}
type wxFactory struct {
factory
}
func (wx wxFactory) create() payment {
return wxPayment{}
}
func main2() {
var ali_factory aliFactory
{
ali_payment := ali_factory.create()
ali_payment.pay("100")
}
var wx_factory wxFactory
{
wx_payment := wx_factory.create()
wx_payment.pay("10")
}
}
四、工厂方法模式的优缺点
优点
- 在工厂方法模式中,工厂方法用来创建客户所需要的产品,同时还向客户隐藏了哪种具体产品类将被实例化这一细节,用户只需要关心所需产品对应的工厂,无须关心创建细节,甚至无须知道具体产品类的类名。
- 基于工厂角色和产品角色的多态性设计是工厂方法模式的关键。它能够使工厂可以自主确定创建何种产品对象,而如何创建这个对象的细节则完全封装在具体工厂内部。
- 使用工厂方法模式的另一个优点是在系统中加入新产品时,无须修改抽象工厂和抽象产品提供的接口,无须修改客户端,也无须修改其他的具体工厂和具体产品,而只要添加一个具体工厂和具体产品就可以了。这样,系统的可扩展性也就变得非常好,完全符合“开闭原则”。
缺点
- 在添加新产品时,需要编写新的具体产品类,而且还要提供与之对应的具体工厂类,系统中类的个数将成对增加,在一定程度上增加了系统的复杂度,有更多的类需要编译和运行,会给系统带来一些额外的开销。
- 由于考虑到系统的可扩展性,需要引入抽象层,在客户端代码中均使用抽象层进行定义,增加了系统的抽象性和理解难度,且在实现时可能需要用到DOM、反射等技术,增加了系统的实现难度。
五、工厂方法模式的应用场景
一个类不知道它所需要的对象的类
:在工厂方法模式中,客户端不需要知道具体产品类的类名,只需要知道所对应的工厂即可,具体的产品对象由具体工厂类创建;客户端需要知道创建具体产品的工厂类。一个类通过其子类来指定创建哪个对象
:在工厂方法模式中,对于抽象工厂类只需要提供一个创建产品的接口,而由其子类来确定具体要创建的对象,利用面向对象的多态性和里氏代换原则,在程序运行时,子类对象将覆盖父类对象,从而使得系统更容易扩展。将创建对象的任务委托给多个工厂子类中的某一个,客户端在使用时可以无须关心是哪一个工厂子类创建产品子类,需要时再动态指定,可将具体工厂类的类名存储在配置文件或数据库中。
六、对比
工厂方法模式和简单工厂模式的区别
简单工厂模式:
- 工厂类负责创建的对象比较少,由于创建的对象较少,不会造成工厂方法中的业务逻辑太过复杂。
- 客户端只知道传入工厂类的参数,对于如何创建对象并不关心。
工厂方法模式:
- 客户端不知道它所需要的对象的类。
- 抽象工厂类通过其子类来指定创建哪个对象。
七、总结
工厂方法模式又称为工厂模式,它属于类创建型模式。在工厂方法模式中,工厂父类负责定义创建产品对象的公共接口,而工厂子类则负责生成具体的产品对象,这样做的目的是将产品类的实例化操作延迟到工厂子类中完成,即通过工厂子类来确定究竟应该实例化哪一个具体产品类。
工厂方法模式包含四个角色:
抽象产品是定义产品的接口,是工厂方法模式所创建对象的超类型,即产品对象的共同父类或接口;
具体产品实现了抽象产品接口,某种类型的具体产品由专门的具体工厂创建,它们之间往往一一对应;
抽象工厂中声明了工厂方法,用于返回一个产品,它是工厂方法模式的核心,任何在模式中创建对象的工厂类都必须实现该接口;
具体工厂是抽象工厂类的子类,实现了抽象工厂中定义的工厂方法,并可由客户调用,返回一个具体产品类的实例。
工厂方法模式是简单工厂模式的进一步抽象和推广。由于使用了面向对象的多态性,工厂方法模式保持了简单工厂模式的优点,而且克服了它的缺点。在工厂方法模式中,核心的工厂类不再负责所有产品的创建,而是将具体创建工作交给子类去做。这个核心类仅仅负责给出具体工厂必须实现的接口,而不负责产品类被实例化这种细节,这使得工厂方法模式可以允许系统在不修改工厂角色的情况下引进新产品。
工厂方法模式的主要优点是增加新的产品类时无须修改现有系统,并封装了产品对象的创建细节,系统具有良好的灵活性和可扩展性;其缺点在于增加新产品的同时需要增加新的工厂,导致系统类的个数成对增加,在一定程度上增加了系统的复杂性。
工厂方法模式适用情况包括:
一个类不知道它所需要的对象的类;
一个类通过其子类来指定创建哪个对象;将创建对象的任务委托给多个工厂子类中的某一个,客户端在使用时可以无须关心是哪一个工厂子类创建产品子类,需要时再动态指定。