Python设计模式详解之5 —— 原型模式

news/2024/11/20 19:40:52/

Prototype 设计模式是一种创建型设计模式,它通过复制已有的实例来创建新对象,而不是通过从头实例化。这种模式非常适合对象的创建成本较高或者需要避免复杂的构造过程时使用。Prototype 模式提供了一种通过克隆来快速创建对象的方式。

1. Prototype 模式简介

Prototype 模式通过定义一个接口来克隆自身,使得客户端代码可以通过复制原型来创建新对象。Python 中,Prototype 模式可以使用内置的 copy 模块来实现浅拷贝或深拷贝。

适用场景
  • 对象创建代价高:创建对象的代价高昂或复杂。
  • 减少对象构建时间:避免复杂的初始化或设置过程。
  • 保留对象状态:创建一个与已有对象状态相同的新对象。

2. Prototype 模式的结构

Prototype 模式通常包含以下角色:

  • Prototype(原型接口):定义一个克隆自身的方法。
  • ConcretePrototype(具体原型):实现克隆方法的类。
  • Client(客户端):通过调用克隆方法来创建新对象。

3. Python 中的 Prototype 模式实现

在 Python 中,可以使用 copy 模块中的 copy()deepcopy() 来实现浅拷贝和深拷贝。

示例:实现 Prototype 模式
python">import copy# Prototype interface
class Prototype:def clone(self):return copy.copy(self)  # 浅拷贝def deep_clone(self):return copy.deepcopy(self)  # 深拷贝# ConcretePrototype class
class Car(Prototype):def __init__(self, make, model, options=None):self.make = makeself.model = modelself.options = options if options is not None else []def __str__(self):return f"Car(make={self.make}, model={self.model}, options={self.options})"# Client code
car1 = Car("Tesla", "Model S", ["Sunroof", "Leather seats"])
car2 = car1.clone()
car3 = car1.deep_clone()# 修改 car2 的选项,不影响原始对象 car1
car2.options.append("GPS")
car3.options.append("Heated seats")print("Original Car:", car1)  # 输出: Car(make=Tesla, model=Model S, options=['Sunroof', 'Leather seats', 'GPS'])
print("Shallow Cloned Car:", car2)  # 输出: Car(make=Tesla, model=Model S, options=['Sunroof', 'Leather seats', 'GPS'])
print("Deep Cloned Car:", car3)  # 输出: Car(make=Tesla, model=Model S, options=['Sunroof', 'Leather seats', 'Heated seats'])

4. 浅拷贝与深拷贝

  • 浅拷贝:创建一个新对象,复制原对象的基本数据类型,但对于引用类型只复制引用,不复制对象本身。
  • 深拷贝:创建一个新对象,递归复制原对象及其引用的对象,生成完全独立的副本。
区别示例:
python">import copylist1 = [[1, 2, 3], [4, 5, 6]]
shallow_copy = copy.copy(list1)
deep_copy = copy.deepcopy(list1)# 修改 shallow_copy 的内部列表
shallow_copy[0].append(100)print("Original List:", list1)  # 输出: [[1, 2, 3, 100], [4, 5, 6]]
print("Shallow Copy:", shallow_copy)  # 输出: [[1, 2, 3, 100], [4, 5, 6]]
print("Deep Copy:", deep_copy)  # 输出: [[1, 2, 3], [4, 5, 6]]

5. Prototype 模式的优缺点

优点

  • 减少对象创建时间:通过复制现有对象,减少了初始化对象的时间。
  • 简化对象构建:避免复杂的初始化逻辑。
  • 保持对象一致性:能够克隆包含复杂结构的对象。

缺点

  • 潜在的深拷贝开销:深拷贝在涉及嵌套引用时可能会带来性能开销。
  • 复杂的拷贝逻辑:需要处理对象中引用的克隆问题,尤其是当对象中包含多个引用类型时。

6. 应用场景

  • 缓存和克隆:在需要缓存对象以备重复使用时,通过克隆实现对象的重复创建。
  • 游戏开发:创建角色或物品时,通过克隆来实现快速复制对象。
  • 原型管理器:通过一个管理器类来存储不同的原型,并根据需要进行克隆。
示例:原型管理器
python">class PrototypeManager:def __init__(self):self._prototypes = {}def register_prototype(self, name, prototype):self._prototypes[name] = prototypedef unregister_prototype(self, name):del self._prototypes[name]def clone(self, name):if name in self._prototypes:return self._prototypes[name].clone()raise ValueError(f"Prototype {name} not found")# 注册和克隆示例
manager = PrototypeManager()
car_prototype = Car("BMW", "X5")
manager.register_prototype("Luxury SUV", car_prototype)cloned_car = manager.clone("Luxury SUV")
print(cloned_car)  # 输出: Car(make=BMW, model=X5, options=[])

7. 总结

Prototype 模式在 Python 中通过 copy 模块实现,提供了基于现有对象的快速创建方式,避免了复杂的构造逻辑。它适用于需要频繁创建对象或保持对象状态一致性的场景,尤其在需要提高对象创建性能时非常有用。


http://www.ppmy.cn/news/1548551.html

相关文章

Mac 修改默认jdk版本

当前会话生效 这里演示将 Java 17 版本降低到 Java 8 查看已安装的 Java 版本: 在终端(Terminal)中运行以下命令,查看已安装的 Java 版本列表 /usr/libexec/java_home -V设置默认 Java 版本: 找到 Java 8 的安装路…

鸿蒙NEXT开发案例:计数器

【引言】(完整代码在最后面) 本文将通过一个简单的计数器应用案例,介绍如何利用鸿蒙NEXT的特性开发高效、美观的应用程序。我们将涵盖计数器的基本功能实现、用户界面设计、数据持久化及动画效果的添加。 【环境准备】 电脑系统&#xff1…

STM32单片机CAN总线汽车线路通断检测-分享

目录 目录 前言 一、本设计主要实现哪些很“开门”功能? 二、电路设计原理图 1.电路图采用Altium Designer进行设计: 2.实物展示图片 三、程序源代码设计 四、获取资料内容 前言 随着汽车电子技术的不断发展,车辆通信接口在汽车电子控…

openeuler设置IP

1 编辑网络配置文件:通常在/etc/sysconfig/network-scripts/目录下,对应的网络接口配置文件名为ifcfg-,例如ifcfg-eth0。 2. 修改配置文件:将BOOTPROTO的值改为static,并设置IPADDR、NETMASK、GATEWAY和DNS等参数。…

Clip结合Faiss+Flask简易版文搜图服务

一、实现 使用目录结构&#xff1a; templates ---upload.html faiss_app.py 前端代码&#xff1a;upload.html <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content&quo…

【企业级分布式系统】ELK优化

文章目录 Elasticsearch作为日志存储时的优化优化ES索引设置优化线程池配置锁定内存&#xff0c;不让JVM使用Swap减少分片数、副本数 Elasticsearch作为日志存储时的优化 linux内核优化、JVM优化、ES配置优化、架构优化&#xff08;filebeat/fluentd代替logstash、加入kafka做…

shell编程规范和脚本变量

什么是shell 人和计算机内核之间的中介&#xff1a; 计算机的语言是二进制&#xff0c;把人类的语言翻译成计算机能够识别的语言&#xff0c;然后让内核来处理 内核完成之后要把结果反馈给用户&#xff0c;要把计算机的翻译成人类能够识别的语言 命令解释器&#xff0c;pyc…

LeetCode40:组合总数II

题目链接&#xff1a;40. 组合总和 II - 力扣&#xff08;LeetCode&#xff09; 代码如下&#xff1a; class Solution { private:vector<vector<int>> result;vector<int> path;void backtracking(vector<int>& candidates, int target, int su…