设计模式中的原型模式在Python程序中的应用示例
原型模式:
原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象。
原型模式本质就是克隆对象,所以在对象初始化操作比较复杂的情况下,很实用,能大大降低耗时,提高性能,因为“不用重新初始化对象,而是动态地获得对象运行时的状态”。
应用特性:
需要大量的基于某个基础原型进行微量修改而得到新原型时使用。
结构特性:
对象的复制机制,即浅复制和深复制。
例1:
#!/usr/bin/envpython
#encoding:utf-8
fromcopyimportcopy,deepcopy
classtest_obj:
def__init__(self,id):
self.id=id
classproto_type:
def__init__(self,name,id):
self.name=name
self.obj=test_obj(id)
defdisplay(self):
printself.name
printself.obj.id
defclone(self):
returncopy(self)
defdeep_clone(self):
returndeepcopy(self)
if'__main__'==__name__:
obj1=proto_type('name1',1)
obj2=obj1.clone()
obj3=obj1.deep_clone()
obj2.name='name2'
obj2.obj.id=2
obj3.name='name3'
obj3.obj.id=3
obj1.display()
obj2.display()
obj3.display()
printobj1.__class__
printobj2.__class__
printobj3.__class__
结果:
name1 2#因为obj2是浅复制,所以对象没有被复制,导致新对象的修改影响了原来的就对象的值 name2 2 name3 3#因为是深复制,所以不会影响之前的旧对象 __main__.proto_type __main__.proto_type __main__.proto_type
这里我们再来回顾一下Python编程基础中关于浅拷贝和深拷贝的知识点:
浅拷贝(ShallowCopy):
指对象的字段被拷贝,而字段引用的对象不会被拷贝,拷贝的对象和源对象只是名称相同,但是他们共用一个实体。
深拷贝(deepcopy):
对对象实例中字段引用的对象也进行拷贝。
好了,基于以上,我们再来看一个例子:
例2:
#encoding=utf-8
#
#bypanda
#原型模式
importcopy
defprintInfo(info):
printunicode(info,'utf-8').encode('gbk')
#拷贝接口
classICloneable:
defshallowClone(self):
returncopy.copy(self)
defdeepClone(self):
returncopy.deepcopy(self)
#工作经历
classWorkExperience(ICloneable):
workData=""
company=""
pass
#简历
classResume(ICloneable):
name=""
sex='未知'
age=0
work=None
def__init__(self,name,work=WorkExperience()):
self.name=name
self.work=work;
defsetPersonInfo(self,sex,age):
self.sex=sex
self.age=age
defsetWorkExperience(self,workData,company):
self.work.workData=workData
self.work.company=company
defdisplay(self):
printInfo('%s,%s,%d'%(self.name,self.sex,self.age))
printInfo('%s,%s'%(self.work.workData,self.work.company))
defclientUI():
a=Resume('大鸟')
a.setPersonInfo('男',29)
a.setWorkExperience("1998-2000","XX公司")
#浅拷贝
b=a.shallowClone()
b.setWorkExperience("2000-2006","YY公司")
#深拷贝
c=a.deepClone()
c.setWorkExperience("2006-2009","ZZ公司")
b.display()
a.display()
c.display()
return
if__name__=='__main__':
clientUI();