Bootstrap

设计模式Python版 原型模式


前言

GOF设计模式分三大类:

  • 创建型模式:关注对象的创建过程,包括单例模式、简单工厂模式、工厂方法模式、抽象工厂模式、原型模式和建造者模式。
  • 结构型模式:关注类和对象之间的组合,包括适配器模式、桥接模式、组合模式、装饰模式、外观模式、享元模式和代理模式。
  • 行为型模式:关注对象之间的交互,包括职责链模式、命令模式、解释器模式、迭代器模式、中介者模式、备忘录模式、观察者模式、状态模式、策略模式、模板方法模式和访问者模式。

一、原型模式

原型模式(Prototype Pattern)

  • 定义:使用原型实例指定创建对象的种类,并且通过克隆这些原型创建新的对象。
  • 解决问题:如何通过克隆来得到一个一模一样的对象?(通过克隆方法所创建的对象是全新的对象,它们在内存中拥有新的地址。)
  • 使用场景:
    • 创建新对象成本较大
    • 系统要保存对象的状态
    • 在运行时动态地创建和定制对象
  • 组成:
    • 原型接口(Prototype):声明克隆自身的接口
    • 具体原型(ConcretePrototype):实现原型接口,提供克隆自身的操作。
    • 客户端(Client):让一个原型克隆自身,从而创建一个新的对象。
  • 优点:
    • 当创建新的对象实例较为复杂时,使用原型模式可以简化对象的创建过程,通过复制一个已有实例可以提高新实例的创建效率
    • 可以使用深克隆的方式保存对象的状态。使用原型模式将对象复制一份并将其状态保存起来,以便在需要的时候使用,例如恢复到某一历史状态,可辅助实现撤销操作
  • 缺点:
    • 注意深克隆与浅克隆的区别和使用场景

在这里插入图片描述

二、原型模式示例

使用原型模式快速创建工作周报

import copy

class Prototype:
    """原型接口"""

    def clone(self):
        # 这里用深拷贝,这意味着对象及其所有引用的对象都将被复制
        # 若不希望复制对象内部的引用类型,copy.copy()来执行浅拷贝
        return copy.deepcopy(self)


class WeeklyLog(Prototype):
    """具体原型"""

    def __init__(self, name=None, date=None, content=None):
        self.name = name
        self.date = date
        self.content = content

客户端代码

def display_log(log: WeeklyLog):
    print("###周报###")
    print(f"姓名:{log.name}")
    print(f"周次:{log.date}")
    print(f"内容:{log.content}")
    print("#" * 10, "\n")


log_lastweek = WeeklyLog("张三", "第12周", "这周工作很忙,每天加班!")
log_thisweek = log_lastweek.clone()
log_thisweek.date = "第13周"
display_log(log_lastweek)
display_log(log_thisweek)
print(log_lastweek is log_thisweek)  # 输出 False,表示是不同的对象

输出结果

###周报###
姓名:张三
周次:第12周
内容:这周工作很忙,每天加班!
##########

###周报###
姓名:张三
周次:第13周
内容:这周工作很忙,每天加班!
##########

False

三、原型管理器

  • 原型管理器(Prototype Manager)是将多个原型对象存储在一个集合中供客户端使用,它是一个专门负责克隆对象的工厂
  • 原型管理器采用单例模式,能节省系统资源,也能更好地管理。定义一个集合存储原型对象,需要某个原型对象的一个克隆,调用相应对象的克隆方法来获得。
import copy

"""原型接口"""

class OfficialDocument:
    def clone(self):
        return copy.deepcopy(self)

    def display(self):
        raise NotImplementedError


"""具体原型"""

class FAR(OfficialDocument):
    def display(self):
        print("《可行性分析报告》")


class SRS(OfficialDocument):
    def display(self):
        print("《软件需求规格说明书》")


"""原型管理器(使用模块单例模式)"""

class PrototypeManager:
    def __init__(self):
        self.official_docs: dict[str, OfficialDocument] = {}
        self.add_official_docs()

    def add_official_docs(self):
        self.official_docs["far"] = FAR()
        self.official_docs["srs"] = SRS()

    def get_official_doc(self, key: str) -> OfficialDocument:
        official_doc = self.official_docs.get(key, None)
        if official_doc:
            return official_doc.clone()


prototype_manager = PrototypeManager()
  • 客户端代码
from prototypes import prototype_manager

doc1 = prototype_manager.get_official_doc("far")
doc2 = prototype_manager.get_official_doc("far")
doc1.display()
doc2.display()
print(doc1 is doc2)

doc3 = prototype_manager.get_official_doc("srs")
doc4 = prototype_manager.get_official_doc("srs")
doc3.display()
doc4.display()
print(doc3 is doc4)
  • 输出结果
《可行性分析报告》
《可行性分析报告》    
False
《软件需求规格说明书》
《软件需求规格说明书》
False

您正在阅读的是《设计模式Python版》专栏!关注不迷路~

;