【C++设计模式】第五篇:原型模式(Prototype)
注意:复现代码时,确保 VS2022 使用 C++17/20 标准以支持现代特性。
克隆对象的效率革命
1. 模式定义与用途
核心思想
- 原型模式:通过复制现有对象(原型)来创建新对象,而非通过new构造。
- 关键用途:
1.减少初始化开销:适用于创建成本高的对象(如数据库连接)。
2.动态配置对象:运行时通过克隆生成预设配置的实例。
经典场景
- 游戏开发:批量生成相同属性的敌人或道具。
- 文档编辑:复制带格式的文本段落。
2. 模式结构解析
UML类图
+---------------------+ +---------------------+
| Prototype | | Client |
+---------------------+ +---------------------+
| + clone(): Prototype|<>------->| - prototype: Prototype
+---------------------+ +---------------------+ ^ |
+---------------------+
| ConcretePrototype |
+---------------------+
| + clone() |
+---------------------+
角色说明
Prototype:抽象接口,声明克隆方法clone()。ConcretePrototype:具体原型类,实现克隆逻辑。Client:通过调用 clone() 创建新对象。
3. 简单示例:基础克隆实现
#include <memory> // 抽象原型接口
class Enemy {
public: virtual ~Enemy() = default; virtual std::unique_ptr<Enemy> clone() const = 0; virtual void attack() const = 0;
}; // 具体原型:骷髅战士
class SkeletonWarrior : public Enemy {
public: std::unique_ptr<Enemy> clone() const override { return std::make_unique<SkeletonWarrior>(*this); // 调用拷贝构造函数 } void attack() const override { std::cout << "骷髅战士挥舞骨刀!\n"; }
}; // 使用克隆
auto original = std::make_unique<SkeletonWarrior>();
auto clone = original->clone();
clone->attack(); // 输出:骷髅战士挥舞骨刀!
4. 完整代码:原型管理器与深拷贝优化
场景:游戏敌人原型注册与批量生成
#include <iostream>
#include <unordered_map>
#include <memory>
#include <string> // 抽象敌人原型
class Enemy {
public: virtual ~Enemy() = default; virtual std::unique_ptr<Enemy> clone() const = 0; virtual void spawn() const = 0; virtual void setHealth(int health) = 0;
}; // 具体原型:火焰恶魔
class FireDemon : public Enemy {
public: FireDemon(int health, const std::string& color) : health_(health), color_(color) {} std::unique_ptr<Enemy> clone() const override { return std::make_unique<FireDemon>(*this); } void spawn() const override { std::cout << "生成" << color_ << "火焰恶魔(生命值:" << health_ << ")\n"; } void setHealth(int health) override { health_ = health; } private: int health_; std::string color_;
}; // 原型管理器(注册表)
class PrototypeRegistry {
public: void registerPrototype(const std::string& key, std::unique_ptr<Enemy> prototype) { prototypes_[key] = std::move(prototype); } std::unique_ptr<Enemy> createEnemy(const std::string& key) { auto it = prototypes_.find(key); if (it != prototypes_.end()) { return it->second->clone(); } return nullptr; } private: std::unordered_map<std::string, std::unique_ptr<Enemy>> prototypes_;
}; // 客户端代码
int main() { PrototypeRegistry registry; // 注册原型 registry.registerPrototype("red_demon", std::make_unique<FireDemon>(100, "红色")); registry.registerPrototype("blue_demon", std::make_unique<FireDemon>(80, "蓝色")); // 批量生成敌人 auto enemy1 = registry.createEnemy("red_demon"); auto enemy2 = registry.createEnemy("blue_demon"); auto enemy3 = registry.createEnemy("red_demon"); enemy1->spawn(); // 生成红色火焰恶魔(生命值:100) enemy2->spawn(); // 生成蓝色火焰恶魔(生命值:80) enemy3->setHealth(50); enemy3->spawn(); // 生成红色火焰恶魔(生命值:50)
}
5. 优缺点分析
| 优点 | 缺点 |
|---|---|
| 避免重复初始化复杂对象 | 需正确处理深拷贝(尤其含指针成员时) |
| 动态添加/删除原型配置 | 每个类需实现克隆方法,增加代码量 |
| 与工厂模式结合扩展性强 | 对简单对象可能得不偿失 |
6. 调试与优化策略
调试技巧(VS2022)
1.深拷贝验证:
- 在拷贝构造函数中设置断点,观察成员变量是否被正确复制。
- 使用 内存断点 检测指针成员是否被重复释放。
2.原型注册表检查:
- 输出注册表的键列表,确认原型是否成功注册。
for (const auto& pair : prototypes_) { std::cout << "已注册原型: " << pair.first << "\n";
}
性能优化
1.原型预初始化:
- 在程序启动时预加载常用原型,减少运行时开销。
2.浅拷贝优化:
- 对只读数据成员使用浅拷贝(需确保生命周期安全)。
class CheapToCopyEnemy : public Enemy {
private: const Texture* sharedTexture_; // 只读资源,浅拷贝
};
相关文章:
【C++设计模式】第五篇:原型模式(Prototype)
注意:复现代码时,确保 VS2022 使用 C17/20 标准以支持现代特性。 克隆对象的效率革命 1. 模式定义与用途 核心思想 原型模式:通过复制现有对象(原型)来创建新对象,而非通过new构造。关键用…...
深入 Vue.js 组件开发:从基础到实践
深入 Vue.js 组件开发:从基础到实践 Vue.js 作为一款卓越的前端框架,其组件化开发模式为构建高效、可维护的用户界面提供了强大支持。在这篇博客中,我们将深入探讨 Vue.js 组件开发的各个方面,从基础概念到高级技巧,助…...
maven导入spring框架
在eclipse导入maven项目, 在pom.xml文件中加入以下内容 junit junit 3.8.1 test org.springframework spring-core ${org.springframework.version} org.springframework spring-beans ${org.springframework.version} org.springframework spring-context ${org.s…...
数据守护者:备份文件的重要性与自动化实践策略
在数字化浪潮席卷全球的今天,数据已成为企业运营和个人生活中不可或缺的核心资源。无论是企业的财务报表、客户资料,还是个人的家庭照片、工作文档,这些数据都承载着巨大的价值。然而,数据丢失的风险无处不在,硬件故障…...
MyBatis @Param 注解详解:指定的参数找不到?
MyBatis Param 注解详解 1. Param 注解的作用 Param 注解用于显式指定方法参数的名称,让 MyBatis 在 SQL 映射文件(XML)或注解中通过该名称访问参数。 核心场景: 方法有多个参数时,避免参数名丢失或混淆。参数为简单…...
【项目日记(八)】内存回收与联调
前言 我们前面实现了三层缓存申请的过程,并完成了三层缓存申请过程的联调!本期我们来介绍三层的缓存的回收机制以及三层整体联调释放的过程。 目录 前言 一、thread cache 回收内存 二、central cache 回收内存 • 如何确定一个对象对应的span • …...
性能测试监控工具jmeter+grafana
1、什么是性能测试监控体系? 为什么要有监控体系? 原因: 1、项目-日益复杂(内部除了代码外,还有中间件,数据库) 2、一个系统,背后可能有多个软/硬件组合支撑,影响性能的因…...
016.3月夏令营:数理类
016.3月夏令营:数理类: 中国人民大学统计学院: http://www.eeban.com/forum.php?modviewthread&tid386109 北京大学化学学院第一轮: http://www.eeban.com/forum.php?m ... 6026&extrapage%3D1 香港大学化学系夏令营&a…...
CS144 Lab Checkpoint 0: networking warm up
Set up GNU/Linux on your computer 我用的是Ubuntu,按照指导书上写的输入如下命令安装所需的软件包: sudo apt update && sudo apt install git cmake gdb build-essential clang \ clang-tidy clang-format gcc-doc pkg-config glibc-doc tc…...
靶场之路-VulnHub-DC-6 nmap提权、kali爆破、shell反连
靶场之路-VulnHub-DC-6 一、信息收集 1、扫描靶机ip 2、指纹扫描 这里扫的我有点懵,这里只有两个端口,感觉是要扫扫目录了 nmap -sS -sV 192.168.122.128 PORT STATE SERVICE VERSION 22/tcp open ssh OpenSSH 7.4p1 Debian 10deb9u6 (protoc…...
给没有登录认证的web应用添加登录认证(openresty lua实现)
这阵子不是deepseek火么?我也折腾了下本地部署,ollama、vllm、llama.cpp都弄了下,webui也用了几个,发现nextjs-ollama-llm-ui小巧方便,挺适合个人使用的。如果放在网上供多人使用的话,得接入登录认证才好&a…...
3月5日作业
代码作业: #!/bin/bash# 清空目录函数 safe_clear_dir() {local dir"$1"local name"$2"if [ -d "$dir" ]; thenwhile true; doread -p "检测到 $name 目录已存在,请选择操作: 1) 清空目录内容 2) 保留目…...
【MySQL】增删改查
目录 一、新增(Create) 单行数据 全列插入 多行数据 指定列插入 插入时间 二、查询(Retrieve) 全列查询 指定列查询 查询字段为表达式 别名 去重:DISTINCT 排序:ORDER BY 条件查询࿱…...
【三维生成】StarGen:基于视频扩散模型的可扩展的时空自回归场景生成
标题:《StarGen: A Spatiotemporal Autoregression Framework with Video Diffusion Model for Scalable and Controllable Scene Generation》 项目:https://zju3dv.github.io/StarGen 来源:商汤科技、浙大CAD、Tetras.AI 文章目录 摘要一、…...
线反转法实现矩形键盘按键识别
由于行、列线为多键共用,各按键彼此将相互发 生影响,必须将行、列线信号配合起来并作适当的处 理,才能确定闭合键的位置。 线反转法 第1步:列线输出为全低电平,则行线中电平由高变低 的所在行为按键所在行。 第2步&…...
在 Element Plus 的 <el-select> 组件中,如果需要将 <el-option> 的默认值设置为 null。 用于枚举传值
文章目录 引言轻松实现 `<el-option>` 的默认值为 `null`I 实现方式监听清空事件 【推荐】使用 v-model 绑定 null添加一个值为 null 的选项处理 null 值的显示引言 背景:接口签名规则要求空串参与,空对象不参与签名计算 // 空字符串“” 参与签名组串,null不参与签…...
大白话面试中应对自我介绍
在面试中,自我介绍是开场的关键环节,它就像你递给面试官的一张“个人名片”,要让面试官快速了解你并对你产生兴趣。下面详细讲讲应对自我介绍的要点及回答范例。 一、自我介绍的时间把控 一般面试中的自我介绍控制在1 - 3分钟比较合适。时间…...
Pytorch构建LeNet进行MNIST识别 #自用
LeNet是一种经典的卷积神经网络(CNN)结构,由Yann LeCun等人在1998年提出,主要用于手写数字识别(如MNIST数据集)。作为最早的实用化卷积神经网络,LeNet为现代深度学习模型奠定了基础,…...
元宇宙崛起:区块链与金融科技共绘数字新世界
文章目录 一、引言二、元宇宙与区块链的深度融合三、区块链在元宇宙金融中的应用四、金融科技在元宇宙中的创新应用五、面临的挑战与机遇《区块链与金融科技》亮点内容简介获取方式 一、引言 随着科技的飞速发展,元宇宙概念逐渐走进人们的视野,成为数字…...
React Native 实现滑一点点内容区块指示器也滑一点点
效果图如上,内容滑一点点,指示器也按比例话一点点,列表宽度跟数据有关。 实现思路如下: 1.监听列表滑动事件,获取列表横向滑动距离,假设为A; 2.获取列表的宽度,及列表可滑动的宽度…...
Three.js 3D地图实战:从GeoJSON数据到交互式可视化(附完整代码)
Three.js 3D地图实战:从GeoJSON数据到交互式可视化 当我们需要在网页上展示一个具有真实地理特征的3D地图时,Three.js无疑是最强大的工具之一。它不仅能让地图以立体的形式呈现,还能添加各种交互效果,让数据可视化变得更加生动。本…...
ThinkPad装Win10企业版后,手把手教你用PowerShell搞定Lenovo Vantage(附依赖包下载)
ThinkPad安装Win10企业版后手动部署Lenovo Vantage的完整指南 当你在ThinkPad上安装了纯净的Windows 10企业版系统后,可能会发现无法通过常规方式安装Lenovo Vantage这款官方管理工具。本文将详细介绍如何通过PowerShell命令手动安装Lenovo Vantage及其所有必需的依…...
轻量级PDF渲染库PdfiumAndroid:Android开发者的高效集成指南
轻量级PDF渲染库PdfiumAndroid:Android开发者的高效集成指南 【免费下载链接】PdfiumAndroid 项目地址: https://gitcode.com/gh_mirrors/pd/PdfiumAndroid 核心价值:为什么选择PdfiumAndroid? 📌 解决PDF渲染痛点&#…...
终极指南:OpCore Simplify如何让你零基础打造完美黑苹果系统
终极指南:OpCore Simplify如何让你零基础打造完美黑苹果系统 【免费下载链接】OpCore-Simplify A tool designed to simplify the creation of OpenCore EFI 项目地址: https://gitcode.com/GitHub_Trending/op/OpCore-Simplify 还在为复杂的OpenCore EFI配置…...
全网最详细的AI产品经理学习路线,非常详细收藏这一篇就够了
前言 AI产品经理作为一个新兴且热门的职业,不仅需要具备传统产品经理的能力,还需要对AI技术有深入的理解和应用。本学习路线旨在帮助有志于成为AI产品经理的学习者系统地掌握所需的知识和技能。 前排提示,文末有大模型AGI-CSDN独家资料包哦…...
毕业论文神器 2026 降AI率平台推荐:工具对比+最好用AI推荐
2026年真正好用的AI论文降重与改写工具,核心看降重效果、去AI味、格式保留、学术适配四大指标。综合实测,千笔AI、ThouPen、豆包、DeepSeek、Grammarly 是当前最值得推荐的梯队,覆盖从免费到付费、从中文到英文、从文科到理工的全场景需求。 …...
LFM2.5-1.2B-Thinking-GGUF实操手册:自定义system prompt提升领域适配性
LFM2.5-1.2B-Thinking-GGUF实操手册:自定义system prompt提升领域适配性 1. 模型简介与核心优势 LFM2.5-1.2B-Thinking-GGUF是Liquid AI推出的轻量级文本生成模型,专为低资源环境优化设计。该模型采用GGUF格式和llama.cpp运行时,在保持高性…...
5倍效率提升:GIMP批量图像处理插件BIMP全攻略
5倍效率提升:GIMP批量图像处理插件BIMP全攻略 【免费下载链接】gimp-plugin-bimp 项目地址: https://gitcode.com/gh_mirrors/gi/gimp-plugin-bimp 在数字内容创作领域,批量图像处理是提升效率的关键环节。GIMP作为免费开源的图像编辑软件&#…...
手把手教你用modf()和fmod()解决C语言浮点数计算中的常见坑
深入解析C语言浮点数计算:modf()与fmod()的实战应用 浮点数计算在C语言开发中无处不在,从游戏物理引擎到嵌入式传感器数据处理,精确的浮点运算直接关系到程序行为的正确性。然而,许多开发者第一次遭遇浮点数计算误差时,…...
从零开始学习C++ -- 基础知识
C入门基础1.C的第一个程序2.命名空间2.1 namespace的价值2.2 namespace的定义2.3命名空间使用3.C输入&输出4.缺省参数5.函数重载6.引用6.1引用的概念和定义6.2引用的特性6.3引用的使用6.4const引用6.5指针和引用的关系7.inline8.nullptr1.C的第一个程序 #include <iost…...
