设计模式:原型模式

# 原型模式(Prototype Pattern): 使用原型实例指定待创建对象的类型,并且通过复制这个原型来创建新的对象。

#原型模式使用场景
# 1)资源优化场景。类初始化需要消耗非常多的资源,这个资源包括数据,硬件资源等,可通过原型复制避免这些消耗
# 2)通过new产生一个对象需要非常繁琐的数据准备或访问权限,这时可以使用原型模式
# 3)一个对象需要提供给其他对象访问,而且每个调用者可能都需要修改其值时?可以考虑使用原型模式复制多个对象供调用者使用,即保护性拷贝就是深拷贝
# 4)性能和安全要求的场景。

#原型模式的优点
# 1)当创建的对象实例较为复杂的时候,使用原型模式可以简化对象的创建过程!
# 2)扩展性好,由于写原型模式的时候使用了抽象原型类,在客户端进行编程的时候可以将具体的原型类通过配置进行读取。
# 2)可以使用深度克隆来保存对象的状态,使用原型模式进行复制。当你需要恢复到某一时刻就直接跳到。比如我们的idea种就有历史版本,或则SVN中也有这样的操作。

# 原型模式的缺点
# 1)配备克隆方法需要对类的功能进行通盘考虑,这对于全新的类不是很难,但对于已有的类不一定很容易,
# 特别当一个类引用不支持串行化的间接对象,或者引用含有循环结构的时候。
# 2)必须实现 Cloneable 接口。

#原型模式的角色
# 1)抽象原型(Prototype)角色:规定了具体原型对象必须实现的接口(如果要提供深拷贝,则必须具有实现clone的接口实现)
# 2)具体原型(ConcretePrototype):也就是继承原型的具体类,从抽象原型派生而来,是客户程序使用的对象,即被复制的对象,需要实现抽象原型角色所要求的接口。
# 3)客户(Client)角色:使用原型对象的客户程序

# 原型模式的主意事项
# 对象拷贝的时候构造函数是不会执行的,原因在于拷贝是直接在堆中进行,这其实也可以理解,new的时候,JVM要走一趟类加载流程,这个流程非常麻烦,
# 在类加载流程中会调用构造函数,最后生成的对象会放到堆中,而拷贝就是直接拷贝堆中的现成的二进制对象,然后重新一个分配内存块。

# 浅克隆
# 仅仅复制所克隆的对象,而不复制它所引用的对象。
# 被复制对象的所有变量都含有与原来的对象相同的值,而所有的对其他对象的引用仍然指向原来的对象。

# 深克隆
# 把要复制的对象所引用的对象都复制了一遍。
# 那些引用其他对象的变量将指向被复制过的新对象,而不再是原有的那些被引用的对象。

from copy import copy, deepcopy


# 原型抽象类
class Prototype:
    def clone(self):
        pass

    def deep_clone(self):
        pass


# workexperience类
class WorkExperience:
    def __init__(self):
        self.timearea = ''
        self.company = ''

    def set_work_experience(self, timearea, company):
        self.timearea = timearea
        self.company = company


# Resume类
class Resume(Prototype):
    def __init__(self, name):
        self.name = name
        self.work_experience = WorkExperience()

    def clone(self):
        return copy(self)

    def deep_clone(self):
        return deepcopy(self)

    def set_person_info(self, sex, age):
        self.sex = sex
        self.age = age

    def set_work_experience(self, timearea, company):
        self.work_experience.set_work_experience(timearea, company)

    def display(self):
        print(self.name)
        print(self.sex, self.age)
        print('work experience', self.work_experience.timearea, self.work_experience.company)


if __name__ == '__main__':
    obj1 = Resume('Victor')
    obj2 = obj1.clone()
    obj3 = obj1.deep_clone()

    obj1.set_person_info('male', 28)
    obj1.set_work_experience('2001-2003', '毕马威')
    obj2.set_person_info('male', 27)
    obj2.set_work_experience('2004-2006', '上交所')
    obj3.set_person_info('male', 26)
    obj3.set_work_experience('2011-2012', '外资企业')

    obj1.display()
    obj2.display()
    obj3.display()

  

 

 




posted @ 2020-08-25 10:01  Adamanter  阅读(70)  评论(0)    收藏  举报