Python设计模式-工厂方法模式讲解_第1页
Python设计模式-工厂方法模式讲解_第2页
Python设计模式-工厂方法模式讲解_第3页
Python设计模式-工厂方法模式讲解_第4页
Python设计模式-工厂方法模式讲解_第5页
已阅读5页,还剩2页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

1/1Python设计模式:工厂方法模式讲解

工厂方法模式是简单工厂模式的进一步抽象和推广,它不仅保持了简单工厂模式能够向客户隐藏类的实例化过程这一优点,而且还通过多态性克服了工厂类过于复杂且不易于扩展的缺点。在工厂方法模式中,处于核心地位的工厂类不再负责所有产品的创建,而是将具体的创建工作交由子类去完成。工厂方法模式中的核心工厂类经过功能抽象之后,成为了一个抽象的工厂角色,仅负责给出具体工厂子类必须实现的接口,而不涉及哪种产品类应当被实例化这一细节。工厂方法模式的一般性结构如下图所示,图中为了简化只给出了一个产品类和一个工厂类,但在实际系统中通常需要设计多个产品类和多个工厂类。

工厂方法模式的实质是将对象的创建延迟到其子类实现,即由子类根据当前情况动态决定应该实例化哪一个产品类。从上图可以看出,工厂方法模式涉及到抽象工厂角色、具体工厂角色、抽象产品角色和具体产品角色四个参与者。

抽象工厂(Creator)角色是工厂方法模式的核心,它负责定义创建抽象产品对象的工厂方法。抽象工厂不能被外界直接调用,但任何在模式中用于创建产品对象的工厂类都必须实现由它所定义的工厂方法。

具体工厂(ConcreteCreator)角色是工厂方法模式的对外接口,它负责实现创建具体产品对象的内部逻辑。具体工厂与应用密切相关,可以被外界直接调用,创建所需要的产品。

抽象产品(Product)角色是工厂方法模式所创建的所有对象的父类,它负责描述所有具体产品共有的公共接口。

具体产品(ConcreteProduct)角色是工厂方法模式的创建目标,所有创建的对象都是充当这一角色的某个具体类的实例。

抽象工厂角色负责声明工厂方法(factorymethod),用来"生产"抽象产品,以下是抽象工厂的示例性Python代码:

classCreator:

"""抽象工厂角色"""

#创建抽象产品的工厂方法deffactoryMethod(self):pass

具体工厂角色负责创建一个具体产品的实例,并将其返回给调用者。具体工厂是与具体产品相关的,实现时一般常用的做法是为每个具体产品定义一个具体工厂。以下是具体工厂的示例性Python代码:

classConcreteCreator(Creator):

"""具体工厂角色"""

#创建具体产品的工厂方法deffactoryMethod(self):product=ConcreteProduct()returnproduct

抽象产品角色的主要目的是为所有的具体产品提供一个共同的接口,通常只需给出相应的声明就可以了,而不用给出具体的实现。以下是抽象产品类的示例性Python代码:

classProduct:

"""抽象产品角色"""

#所有产品类的公共接口definterface(self):pass

具体产品角色充当最终的创建目标,一般来讲它是抽象产品类的子类,实现了抽象产品类中定义的所有工厂方法,实际应用时通常会具有比较复杂的业务逻辑。以下是具体产品类的示例性Python代码:

classConcreteProduct(Product):

"""具体产品角色"""

#公共接口的.实现definterface(self):print"ConcreteProductMethod"

在应用工厂方法模式时,通常还需要再引入一个客户端角色,由它负责创建具体的工厂对象,然后再调用工厂对象中的工厂方法来创建相应的产品对象。以下是客户端的示例性Python代码:

classClient:

"""客户端角色"""

defrun(self):creator=ConcreteCreator()product=()()#主函数if(__name__=="__main__"):client=Client()()

在这个简单的示意性实现里,充当具体产品和具体工厂角色的类都只有一个,但在真正的实际应用中,通常遇到的都是同时会有多个具体产品类的情况,此时相应地需要提供多个具体工厂类,每个具体工厂都负责生产对应的具体产品。

工厂方法模式的活动序列如下图所示,客户端Client首先创建ConcreteCreator对象,然后调用ConcreteCreator对象的工厂方法factoryMethod(),由它负责"生产"出所需要的ConcreteProduct对象。

下面我们来看一个具体案例:

如果你开一家Pizza店(PizzaStore抽象类)卖各种风味的Pizza(Pizza子类),那么你需要根据客户要求准备相应的Pizza(创建Pizza对象),然后烘烤、切片、包装;

最简单的做法就是在PizzaStore中根据客户要求(类型判断)创建相应的Pizza对象,然后调用Pizza自身(由Pizza抽象类实现)的烘烤、切片和包装方法;

但这样的代码缺乏弹性,因为你让一个抽象类去依赖具体的对象;我们可以创建一个工厂来生产Pizza,根据传入的不同类型值返回不同Pizza对象,即从PizzaStore中将创建对象的代码挪到工厂中。但这只是一个编程技巧,并不算模式。

在工厂方法模式中,我们在PizzaStore中定义一个抽象接口(create_pizza)作为抽象的工厂,而order_pizza是它的客户;将Pizza对象的创建放到PizzaStore子类去解决。

现有Cheese和Clam两款Pizza,以及NY和Chicago两家分店,每家店的同款Pizza的口味不同——为迎合当地口味做了改进,主要差别来自不同的原材料,因此我们实现四个Pizza类型(NYStyleCheesePizza、NYStyleClamPizza、ChicagoStyleCheesePizza和ChicagoStyleClamPizza),每种使用不同的原材料组合,根据客户所在城市和选择款式我们创建不同的对象;根据工厂方法,我们将对象创建的代码放到PizzaStore子类去实现。

代码:

#!/usr/bin/pythonclassPizza:

name=""

dough=""

sauce=""

toppings=

defprepare(self):

print"Preparing%s"%

print"dough:%s"%

print"sauce:%s"%

print"addtoppings:"

fornin:

print"

%s"%n

defbake(self):

print"Bakefor25minutesat350."

defcut(self):

print"Cuttingintodiagonalslices."

defbox(self):

print"Putintoofficialbox."

defget_name(self):

return

classPizzaStore:

deforder_pizza(self,pizza_type):

=_pizza(pizza_type)

()

()

()

()

return

defcreate_pizza(self,pizza_type):

pass

classNYStyleCheesePizza(Pizza):

def__init__(self):

="NYStyleCheesePizza"

="NYDough"

="NYSauce"

("NYtooppingA")

("NYtooppingB")

classChicagoStyleCheesePizza(Pizza):

def__init__(self):

="ChicagoStyleCheesePizza"

="ChicagoDough"

="ChicagoSauce"

("ChicagotooppingA")

defcut(self):

print"Cuttingintosquareslices."

classNYStyleClamPizza(Pizza):

def__init__(self):

="NYStyleClamPizza"

="NYDough"

="NYSauce"

("NYtooppingA")

("NYtooppingB")

classChicagoStyleClamPizza(Pizza):

def__init__(self):

="ChicagoStyleClamPizza"

="ChicagoDough"

="ChicagoSauce"

("ChicagotooppingA")

defcut(self):

print"Cuttingintosquareslices."

classNYPizzaStore(PizzaStore):

defcreate_pizza(self,pizza_type):

ifpizza_type=="cheese":

returnNYStyleCheesePizza()

elifpizza_type=="clam":

returnNYStyleClamPizza()

else:

returnNone

classChicagoPizzaStore(PizzaStore):

defcreate_pizza(self,pizza_type):

ifpizza_type=="cheese":

returnChicagoStyleCheesePizza()

elifpizza_type=="clam":

returnChicagoStyleClamPizza()

else:

returnNone

if__name__=="__main__":

ny_store=NYPizzaStore()

chicago_store=ChicagoPizzaStore()

pizza=ny__pizza("cheese")

print"Mikeordereda%s."%_name()

print

pizza=chicago__pizza("clam")

print"Johnordereda%s."%_name()

print

输出:

PreparingNYStyleCheesePizza

dough:NYDough

sauce:NYSauce

addtoppings:

NYtooppingA

NYtooppingBBakefor25minutesat350.Cuttingintodiagonalslices.Putintoofficialbox.MikeorderedaNYStyleCheesePizza.

PreparingCh

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论