【设计模式】【创建型模式(Creational Patterns)】之原型模式(Prototype Pattern)
1. 设计模式原理说明
原型模式(Prototype Pattern) 是一种创建型设计模式,它允许你通过复制现有对象来创建新对象,而无需通过构造函数来创建。这种方式可以提高性能,尤其是在对象初始化需要消耗大量资源或耗时较长的情况下。原型模式的关键在于对象的克隆方法,通过该方法可以创建一个与原对象具有相同属性的新对象。
主要角色
- Prototype(抽象原型类):声明了克隆自身的接口。
- ConcretePrototype(具体原型类):实现了克隆方法,用于创建新的对象实例。
- Client(客户端):使用原型类创建新对象。
2. UML 类图及解释
UML 类图
+----------------+ +---------------------+
| Prototype | | ConcretePrototype |
|----------------| |---------------------|
| + clone(): Prototype | + clone(): Prototype|
+----------------+ +---------------------++----------------+ |
| Client | |
|----------------| |
| - prototype: Prototype |
| +---------+
| + createClone(): Prototype |
| + mainMethod(): void |
+----------------+ |
类图解释
- Prototype:定义了一个克隆自身的接口。这个接口通常是一个
clone方法,用于创建一个新的对象实例。 - ConcretePrototype:实现了
Prototype接口中定义的clone方法,用于创建新的对象实例。具体的克隆逻辑由具体原型类实现。 - Client:使用原型类创建新对象。客户端持有一个原型对象的引用,并通过调用其
clone方法来创建新的对象实例。
3. 代码案例及逻辑详解
Java 代码案例
// 抽象原型类
interface Prototype {Prototype clone();
}// 具体原型类
class ConcretePrototype implements Prototype {private String value;public ConcretePrototype(String value) {this.value = value;}@Overridepublic Prototype clone() {// 浅拷贝return new ConcretePrototype(this.value);}@Overridepublic String toString() {return "ConcretePrototype [value=" + value + "]";}
}// 客户端
class Client {private Prototype prototype;public Client(Prototype prototype) {this.prototype = prototype;}public Prototype createClone() {return prototype.clone();}public static void main(String[] args) {ConcretePrototype original = new ConcretePrototype("Original Value");Client client = new Client(original);ConcretePrototype clone1 = (ConcretePrototype) client.createClone();ConcretePrototype clone2 = (ConcretePrototype) client.createClone();System.out.println("Original: " + original);System.out.println("Clone 1: " + clone1);System.out.println("Clone 2: " + clone2);}
}
C++ 代码案例
#include <iostream>
#include <string>// 抽象原型类
class Prototype {
public:virtual Prototype* clone() const = 0;virtual ~Prototype() {}
};// 具体原型类
class ConcretePrototype : public Prototype {
private:std::string value;public:ConcretePrototype(const std::string& value) : value(value) {}Prototype* clone() const override {return new ConcretePrototype(*this);}friend std::ostream& operator<<(std::ostream& os, const ConcretePrototype& p) {return os << "ConcretePrototype [value=" << p.value << "]";}
};// 客户端
class Client {
private:Prototype* prototype;public:Client(Prototype* prototype) : prototype(prototype) {}Prototype* createClone() {return prototype->clone();}~Client() {delete prototype;}
};int main() {ConcretePrototype original("Original Value");Client client(&original);ConcretePrototype* clone1 = dynamic_cast<ConcretePrototype*>(client.createClone());ConcretePrototype* clone2 = dynamic_cast<ConcretePrototype*>(client.createClone());std::cout << "Original: " << original << std::endl;std::cout << "Clone 1: " << *clone1 << std::endl;std::cout << "Clone 2: " << *clone2 << std::endl;delete clone1;delete clone2;return 0;
}
Python 代码案例
import copy# 抽象原型类
class Prototype:def clone(self):pass# 具体原型类
class ConcretePrototype(Prototype):def __init__(self, value):self.value = valuedef clone(self):# 使用深拷贝return copy.deepcopy(self)def __str__(self):return f"ConcretePrototype [value={self.value}]"# 客户端
class Client:def __init__(self, prototype):self.prototype = prototypedef create_clone(self):return self.prototype.clone()if __name__ == "__main__":original = ConcretePrototype("Original Value")client = Client(original)clone1 = client.create_clone()clone2 = client.create_clone()print("Original:", original)print("Clone 1:", clone1)print("Clone 2:", clone2)
Go 代码案例
package mainimport ("fmt"
)// 抽象原型类
type Prototype interface {Clone() Prototype
}// 具体原型类
type ConcretePrototype struct {Value string
}func (p *ConcretePrototype) Clone() Prototype {// 浅拷贝return &ConcretePrototype{Value: p.Value}
}func (p *ConcretePrototype) String() string {return fmt.Sprintf("ConcretePrototype [value=%s]", p.Value)
}// 客户端
type Client struct {prototype Prototype
}func NewClient(prototype Prototype) *Client {return &Client{prototype: prototype}
}func (c *Client) CreateClone() Prototype {return c.prototype.Clone()
}func main() {original := &ConcretePrototype{Value: "Original Value"}client := NewClient(original)clone1 := client.CreateClone().(*ConcretePrototype)clone2 := client.CreateClone().(*ConcretePrototype)fmt.Println("Original:", original)fmt.Println("Clone 1:", clone1)fmt.Println("Clone 2:", clone2)
}
4. 总结
原型模式 是一种创建型设计模式,它通过复制现有对象来创建新对象,而不是通过构造函数。这种方式可以提高性能,特别是在对象初始化需要消耗大量资源或耗时较长的情况下。原型模式的主要优点包括:
- 性能提升:通过复制现有对象来创建新对象,可以避免复杂的初始化过程,提高性能。
- 代码简洁:减少了重复的代码,特别是当对象的初始化过程非常复杂时。
- 灵活度高:可以通过修改原型对象的属性来创建不同配置的对象实例。
然而,原型模式也有一些缺点,例如:
- 深拷贝和浅拷贝的问题:需要仔细处理对象的深拷贝和浅拷贝问题,否则可能会导致意外的行为。
- 类必须实现克隆方法:每个需要被克隆的类都必须实现
clone方法,这可能会增加代码的复杂性。
总的来说,原型模式在需要频繁创建相似对象的场景中非常有用,但在简单对象的创建中可能显得过于复杂。选择是否使用原型模式应根据具体的需求和场景来决定。
相关文章:
【设计模式】【创建型模式(Creational Patterns)】之原型模式(Prototype Pattern)
1. 设计模式原理说明 原型模式(Prototype Pattern) 是一种创建型设计模式,它允许你通过复制现有对象来创建新对象,而无需通过构造函数来创建。这种方式可以提高性能,尤其是在对象初始化需要消耗大量资源或耗时较长的情…...
黄仁勋:人形机器人在内,仅有三种机器人有望实现大规模生产
11月23日,芯片巨头、AI时代“卖铲人”和最大受益者、全球市值最高【英伟达】创始人兼CEO黄仁勋在香港科技大学被授予工程学荣誉博士学位;并与香港科技大学校董会主席沈向洋展开深刻对话,涉及人工智能(AI)、计算力、领导…...
【C语言】宏定义详解
C语言中的宏定义(#define)详细解析 在C语言中,宏定义是一种预处理指令,使用 #define 关键字定义。它由预处理器(Preprocessor)在编译前处理,用于定义常量、代码片段或函数样式的代码替换。宏是…...
LangChain——多向量检索器
每个文档存储多个向量通常是有益的。在许多用例中,这是有益的。 LangChain 有一个基础 MultiVectorRetriever ,这使得查询此类设置变得容易。很多复杂性在于如何为每个文档创建多个向量。本笔记本涵盖了创建这些向量和使用 MultiVectorRetriever 的一些常…...
《岩石学报》
本刊主要报道有关岩石学基础理论的岩石学领域各学科包括岩浆岩石学、变质岩石学、沉积岩石学、岩石大地构造学、岩石同位素年代学和同位素地球化学、岩石成矿学、造岩矿物学等方面的重要基础理论和应用研究成果,同时也刊载综述性文章、问题讨论、学术动态以及书评等…...
数据结构 (12)串的存储实现
一、顺序存储结构 顺序存储结构是用一组连续的存储单元来存储串中的字符序列。这种存储方式类似于线性表的顺序存储结构,但串的存储对象仅限于字符。顺序存储结构又可以分为定长顺序存储和堆分配存储两种方式。 定长顺序存储: 使用静态数组存储ÿ…...
职场发展陷阱
一、只有执行,没有思考 二、只有过程,没有结果 三、只有重复,没有精进 四、不懂向上管理 五、定期汇报 六、不要憋大招 七、多同步信息...
Xcode15(iOS17.4)打包的项目在 iOS12 系统上启动崩溃
0x00 启动崩溃 崩溃日志,只有 2 行,看不出啥来。 0x01 默认配置 由于我开发时,使用的 Xcode 14.1,打包在另外一台电脑 Xcode 15.3 Xcode 14.1 Build Settings -> Asset Catalog Compliter - Options Xcode 15.3 Build S…...
极狐GitLab 17.6 正式发布几十项与 DevSecOps 相关的功能【二】
GitLab 是一个全球知名的一体化 DevOps 平台,很多人都通过私有化部署 GitLab 来进行源代码托管。极狐GitLab 是 GitLab 在中国的发行版,专门为中国程序员服务。可以一键式部署极狐GitLab。 学习极狐GitLab 的相关资料: 极狐GitLab 官网极狐…...
PVE相关名词通俗表述方式———多处细节实验(方便理解)
PVE设置初期,对CIDR、 网关、 LinuxBridge、VLAN等很有困惑的朋友一定很需要一篇能够全面通俗易懂的方式去理解PVE 中Linux网桥的工作方式,就像操作一个英雄,多个技能,还是需要一点点去学习理解的,如果你上来就对着别人…...
Ansible--自动化运维工具
Ansible自动化运维工具介绍 1.Ansible介绍 Ansible是一款自动化运维工具,基于Python开发,集合了众多运维工具(puppet、cfengine、chef、func、fabric)的优点,实现了批量系统配置、批量程序部署、批量运行命令等功能。…...
微信小程序学习指南从入门到精通
🗽微信小程序学习指南从入门到精通🗽 🔝微信小程序学习指南从入门到精通🔝✍前言✍💻微信小程序学习指南前言💻一、🚀文章列表🚀二、🔯教程文章的好处🔯1. ✅…...
微服务篇-深入了解使用 RestTemplate 远程调用、Nacos 注册中心基本原理与使用、OpenFeign 的基本使用
🔥博客主页: 【小扳_-CSDN博客】 ❤感谢大家点赞👍收藏⭐评论✍ 文章目录 1.0 认识微服务 1.1 单体架构 1.2 微服务 1.3 SpringCloud 框架 2.0 服务调用 2.1 RestTemplate 远程调用 3.0 服务注册和发现 3.1 注册中心原理 3.2 Nacos 注册中心 …...
使用 Django 构建支持 Kubernetes API 测试连接的 POST 接口
文章目录 使用 Django 构建支持 Kubernetes API 测试连接的 POST 接口功能需求使用 kubectl 获取 Token命令解析输出示例 完整代码实现Kubernetes API 客户端类功能说明 Django 接口视图关键点解析 路由配置 接口测试请求示例响应结果成功错误 优化建议1. 安全性2. 错误处理3. …...
十二、正则表达式、元字符、替换修饰符、手势和对话框插件
1. 正则表达式 1.1 基本使用 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>Document</title&g…...
计算机毕业设计Python+大模型美食推荐系统 美食可视化 美食数据分析大屏 美食爬虫 美团爬虫 机器学习 大数据毕业设计 Django Vue.js
温馨提示:文末有 CSDN 平台官方提供的学长联系方式的名片! 温馨提示:文末有 CSDN 平台官方提供的学长联系方式的名片! 温馨提示:文末有 CSDN 平台官方提供的学长联系方式的名片! 作者简介:Java领…...
【后端面试总结】MySQL索引
数据库索引不只一种实现方法,但是其中最具代表性,也是我们面试中遇到最多的无疑是B树。 索引为什么选择B树 数据量很大的查找,是不能直接放入内存的,而是需要什么数据就通过磁盘IO去获得。 红黑树,AVL树等二叉查找树…...
[蓝桥杯 2021 省 AB2] 小平方
题目描述 小蓝发现,对于一个正整数 nn 和一个小于 nn 的正整数 vv,将 vv 平方后对 nn 取余可能小于 nn 的一半,也可能大于等于 nn 的一半。 请问,在 11 到 n−1n−1 中, 有多少个数平方后除以 nn 的余数小于 nn 的一半。 例如&…...
Jmeter测试工具的安装和使用,mac版本,jmeter版本5.2.1
Jmeter测试工具的安装和使用JSON格式请求 一、安装1、安装jdk包和设置java环境2、去官网下载Jmeter3、解压后,打开mac终端,进入apache-jmeter的bin文件开启jmeter 二、使用jmeter1、添加线程2、添加HTTP请求3、配置请求的协议、IP地址、端口号、请求方法…...
kmeans 最佳聚类个数 | 轮廓系数(越大越好)
轮廓系数越大,表示簇内实例之间紧凑,簇间距离大,这正是聚类的标准概念。 簇内的样本应该尽可能相似。不同簇之间应该尽可能不相似。 目的:鸢尾花数据进行kmeans聚类,最佳聚类个数是多少? plot(iris[,1:4…...
日语AI面试高效通关秘籍:专业解读与青柚面试智能助攻
在如今就业市场竞争日益激烈的背景下,越来越多的求职者将目光投向了日本及中日双语岗位。但是,一场日语面试往往让许多人感到步履维艰。你是否也曾因为面试官抛出的“刁钻问题”而心生畏惧?面对生疏的日语交流环境,即便提前恶补了…...
超短脉冲激光自聚焦效应
前言与目录 强激光引起自聚焦效应机理 超短脉冲激光在脆性材料内部加工时引起的自聚焦效应,这是一种非线性光学现象,主要涉及光学克尔效应和材料的非线性光学特性。 自聚焦效应可以产生局部的强光场,对材料产生非线性响应,可能…...
Prompt Tuning、P-Tuning、Prefix Tuning的区别
一、Prompt Tuning、P-Tuning、Prefix Tuning的区别 1. Prompt Tuning(提示调优) 核心思想:固定预训练模型参数,仅学习额外的连续提示向量(通常是嵌入层的一部分)。实现方式:在输入文本前添加可训练的连续向量(软提示),模型只更新这些提示参数。优势:参数量少(仅提…...
盘古信息PCB行业解决方案:以全域场景重构,激活智造新未来
一、破局:PCB行业的时代之问 在数字经济蓬勃发展的浪潮中,PCB(印制电路板)作为 “电子产品之母”,其重要性愈发凸显。随着 5G、人工智能等新兴技术的加速渗透,PCB行业面临着前所未有的挑战与机遇。产品迭代…...
土地利用/土地覆盖遥感解译与基于CLUE模型未来变化情景预测;从基础到高级,涵盖ArcGIS数据处理、ENVI遥感解译与CLUE模型情景模拟等
🔍 土地利用/土地覆盖数据是生态、环境和气象等诸多领域模型的关键输入参数。通过遥感影像解译技术,可以精准获取历史或当前任何一个区域的土地利用/土地覆盖情况。这些数据不仅能够用于评估区域生态环境的变化趋势,还能有效评价重大生态工程…...
大模型多显卡多服务器并行计算方法与实践指南
一、分布式训练概述 大规模语言模型的训练通常需要分布式计算技术,以解决单机资源不足的问题。分布式训练主要分为两种模式: 数据并行:将数据分片到不同设备,每个设备拥有完整的模型副本 模型并行:将模型分割到不同设备,每个设备处理部分模型计算 现代大模型训练通常结合…...
大学生职业发展与就业创业指导教学评价
这里是引用 作为软工2203/2204班的学生,我们非常感谢您在《大学生职业发展与就业创业指导》课程中的悉心教导。这门课程对我们即将面临实习和就业的工科学生来说至关重要,而您认真负责的教学态度,让课程的每一部分都充满了实用价值。 尤其让我…...
云原生玩法三问:构建自定义开发环境
云原生玩法三问:构建自定义开发环境 引言 临时运维一个古董项目,无文档,无环境,无交接人,俗称三无。 运行设备的环境老,本地环境版本高,ssh不过去。正好最近对 腾讯出品的云原生 cnb 感兴趣&…...
LINUX 69 FTP 客服管理系统 man 5 /etc/vsftpd/vsftpd.conf
FTP 客服管理系统 实现kefu123登录,不允许匿名访问,kefu只能访问/data/kefu目录,不能查看其他目录 创建账号密码 useradd kefu echo 123|passwd -stdin kefu [rootcode caozx26420]# echo 123|passwd --stdin kefu 更改用户 kefu 的密码…...
无人机侦测与反制技术的进展与应用
国家电网无人机侦测与反制技术的进展与应用 引言 随着无人机(无人驾驶飞行器,UAV)技术的快速发展,其在商业、娱乐和军事领域的广泛应用带来了新的安全挑战。特别是对于关键基础设施如电力系统,无人机的“黑飞”&…...
