C++创建型模式之原型模式

C++ 原型模式(Prototype Pattern)

1. 解决的问题

原型模式(Prototype Pattern)是一种创建型设计模式,用于解决对象创建的问题,特别是在需要创建多个相似对象时,避免使用重复的构造代码。原型模式通过复制已有对象(原型)来创建新对象,而不是通过实例化一个类来创建。

2. 适用场景
  • 当系统需要创建多个相似对象,并且这些对象之间的差异只是部分属性值不同。
  • 当对象的创建过程比较复杂,并且通过复制现有对象可以简化创建过程。
  • 当需要避免使用子类的创建方式来生成对象时。
3. 模式的参与者角色
  • Prototype(抽象原型):声明一个克隆自身的接口。
  • ConcretePrototype(具体原型):实现克隆自身的操作。
  • Client(客户端):使用原型对象来克隆新的对象。
4. 示例代码

假设我们正在开发一个复杂的角色扮演游戏(RPG),游戏中有各种不同的角色,这些角色可以是玩家角色(Player)或敌人角色(Enemy)。每个角色都有不同的属性和能力,例如生命值、攻击力、防御力等。为了简化角色的创建过程,我们可以使用原型设计模式来复制现有的角色,并根据需要进行微调。

角色类图
+----------------+
|   Prototype    |
|----------------|
| + clone()      |
| + print()      |
+----------------+^|
+----------------+
|    Player      |
|----------------|
| + clone()      |
| + print()      |
+----------------+^|
+----------------+
|    Enemy       |
|----------------|
| + clone()      |
| + print()      |
+----------------+

代码
#include <iostream>
#include <string>
#include <memory>
#include <vector>// Prototype 接口
class Prototype {
public:virtual ~Prototype() {}virtual std::unique_ptr<Prototype> clone() const = 0;virtual void print() const = 0;virtual void setHealth(int health) = 0;virtual void setAttack(int attack) = 0;virtual void setDefense(int defense) = 0;
};// BaseCharacter 抽象类
class BaseCharacter : public Prototype {
public:BaseCharacter(std::string name, int health, int attack, int defense): name_(name), health_(health), attack_(attack), defense_(defense) {}void setHealth(int health) override { health_ = health; }void setAttack(int attack) override { attack_ = attack; }void setDefense(int defense) override { defense_ = defense; }void print() const override {std::cout << "Name: " << name_ << ", Health: " << health_<< ", Attack: " << attack_ << ", Defense: " << defense_ << std::endl;}protected:std::string name_;int health_;int attack_;int defense_;
};// Player 具体原型
class Player : public BaseCharacter {
public:Player(std::string name, int health, int attack, int defense): BaseCharacter(name, health, attack, defense) {}std::unique_ptr<Prototype> clone() const override {return std::make_unique<Player>(*this);}
};// Enemy 具体原型
class Enemy : public BaseCharacter {
public:Enemy(std::string name, int health, int attack, int defense): BaseCharacter(name, health, attack, defense) {}std::unique_ptr<Prototype> clone() const override {return std::make_unique<Enemy>(*this);}
};// 客户端代码
int main() {// 创建原型对象auto playerPrototype = std::make_unique<Player>("Hero", 100, 20, 15);auto enemyPrototype = std::make_unique<Enemy>("Goblin", 50, 10, 5);// 创建角色列表std::vector<std::unique_ptr<Prototype>> characters;// 克隆玩家角色auto player1 = playerPrototype->clone();player1->setHealth(80); // 微调生命值characters.push_back(std::move(player1));auto player2 = playerPrototype->clone();player2->setHealth(90); // 微调生命值player2->setAttack(25); // 微调攻击力characters.push_back(std::move(player2));// 克隆敌人角色auto enemy1 = enemyPrototype->clone();characters.push_back(std::move(enemy1));auto enemy2 = enemyPrototype->clone();enemy2->setHealth(60); // 微调生命值enemy2->setAttack(15); // 微调攻击力characters.push_back(std::move(enemy2));// 打印所有角色信息for (const auto& character : characters) {character->print();}return 0;
}

代码说明
  1. Prototype 接口定义了克隆和打印方法,以及设置角色属性的方法。
  2. BaseCharacter 是一个抽象类,实现了 Prototype 接口的通用部分,包括角色属性的设置和打印方法。
  3. Player 和 Enemy 是具体原型类,继承自 BaseCharacter,并实现了 clone 方法。
  4. Client 在 main 函数中创建了原型对象,并通过克隆这些原型对象来创建新的角色。客户端可以根据需要微调角色的属性。

总结

通过原型设计模式,我们可以轻松地复制现有角色,避免了重复的构造代码,并且可以根据需要对克隆的角色进行微调。这种模式在复杂的游戏场景中非常有用,特别是在需要创建多个相似角色时。

原型模式与 C++ 拷贝构造函数的相似性与不同点

相似性
  1. 对象复制

    • 原型模式:原型模式的核心思想是通过复制已有对象来创建新对象。
    • 拷贝构造函数:拷贝构造函数用于从已有对象创建一个新对象。
  2. 避免重复构造

    • 原型模式:避免了重复的构造代码,通过复制已有对象来创建新对象。
    • 拷贝构造函数:避免了重复的构造代码,通过复制已有对象来创建新对象。
  3. 对象状态的复用

    • 原型模式:可以复用已有对象的状态来创建新对象。
    • 拷贝构造函数:可以复用已有对象的状态来创建新对象。
不同点
  1. 设计模式 vs. 语言特性

    • 原型模式:是一种设计模式,属于面向对象编程的设计原则之一。
    • 拷贝构造函数:是 C++ 语言的一个特性,用于实现对象的复制。
  2. 接口定义

    • 原型模式:通常定义一个 clone() 方法,用于克隆对象。
    • 拷贝构造函数:是类的构造函数,定义为 ClassName(const ClassName& other),用于从已有对象创建新对象。
  3. 实现方式

    • 原型模式:可以由程序员显式实现 clone() 方法,并在方法内部调用拷贝构造函数或赋值操作符。
    • 拷贝构造函数:由编译器自动生成或由程序员显式实现。
  4. 灵活性

    • 原型模式:提供了更灵活的对象复制机制,可以在运行时选择不同的原型进行复制。
    • 拷贝构造函数:是类的固有特性,无法在运行时选择不同的构造方式。
  5. 使用场景

    • 原型模式:适用于需要创建多个相似对象,并且这些对象之间的差异只是部分属性值不同。
    • 拷贝构造函数:适用于任何需要从已有对象创建新对象的场景。
示例代码
#include <iostream>
#include <string>
#include <memory>// 使用原型模式的类
class Prototype {
public:virtual ~Prototype() {}virtual std::unique_ptr<Prototype> clone() const = 0;virtual void print() const = 0;
};class ConcretePrototype : public Prototype {
public:ConcretePrototype(std::string name) : name_(name) {}std::unique_ptr<Prototype> clone() const override {return std::make_unique<ConcretePrototype>(*this);}void print() const override {std::cout << "ConcretePrototype: " << name_ << std::endl;}private:std::string name_;
};// 使用拷贝构造函数的类
class CopyConstructorExample {
public:CopyConstructorExample(std::string name) : name_(name) {}CopyConstructorExample(const CopyConstructorExample& other) : name_(other.name_) {std::cout << "Copy Constructor Called" << std::endl;}void print() const {std::cout << "CopyConstructorExample: " << name_ << std::endl;}private:std::string name_;
};int main() {// 原型模式示例auto prototype1 = std::make_unique<ConcretePrototype>("Prototype1");auto clone1 = prototype1->clone();clone1->print();// 拷贝构造函数示例CopyConstructorExample example1("Example1");CopyConstructorExample example2 = example1;example2.print();return 0;
}

总结

  • 相似性:原型模式和拷贝构造函数都用于对象的复制,避免了重复的构造代码。
  • 不同点:原型模式是一种设计模式,通过 clone() 方法实现对象复制;拷贝构造函数是 C++ 语言特性,通过 ClassName(const ClassName& other) 实现对象复制。原型模式提供了更灵活的对象复制机制。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.xdnf.cn/news/17345.html

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

leetcode226:反转二叉树

给你一棵二叉树的根节点 root &#xff0c;翻转这棵二叉树&#xff0c;并返回其根节点。 示例 1&#xff1a; 输入&#xff1a;root [4,2,7,1,3,6,9] 输出&#xff1a;[4,7,2,9,6,3,1]示例 2&#xff1a; 输入&#xff1a;root [2,1,3] 输出&#xff1a;[2,3,1]示例 3&#x…

Excel365和WPS中提取字符串的五种方法

一、问题的提出 如何在WPS或者Excel365中提取A列指定的字符串&#xff0c;从"面"开始一直到".pdf"? 问题的提出 二、问题的分析 我们可以采用多种方法解决这个问题&#xff0c;由于A列到B列的提取是非常有规律的&#xff0c;因此我们可以采用如下几种方…

下载jakarta-taglibs-standard-current.zip

官网&#xff1a;https://archive.apache.org/dist/jakarta/taglibs/standard/binaries/ 下载版本&#xff1a;

Qt信号和槽

信号和槽的概念 在Linux中我们也学过信号 Signal&#xff0c;这是进程间通信的一种方式&#xff0c;这里大致分为三个要素&#xff1a; 信号源&#xff1a;谁发送的信号&#xff08;用户进程&#xff0c;系统内核&#xff0c;终端或者作业控制&#xff0c;&#xff09; 信号的类…

MATLAB绘图

一、实验内容和步骤 MATLAB的图形功能非常强大&#xff0c;可以对二维、三维数据用图形表现&#xff0c;并可以对图形的线形、曲面、视觉、色彩和光线等进行处理。 1、绘制二维曲线 绘制如下图所示的图形&#xff0c;把图形窗口分割为2列2行&#xff0c;在窗口1中绘制一条正弦…

H3C NX30Pro刷机教程-2024-11-16

H3C NX30Pro刷机教程-2024-11-16 ref: http://www.ttcoder.cn/index.php/2024/11/03/h3c-nx30pro亲测无需分区备份 路由器-新机初始化设置路由器登录密码telnet进入路由器后台 刷机上传uboot到路由器后台在Windows环境下解压后的软件包中打开 tftpd64.exe在NX30Pro环境下通过以…

boost之property

简介 property在boost.graph中有使用&#xff0c;用于表示点属性或者边属性 结构 #mermaid-svg-56YI0wFLPH0wixrJ {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-56YI0wFLPH0wixrJ .error-icon{fill:#552222;}#me…

[C++] 智能指针

文章目录 智能指针的使用原因及场景分析为什么需要智能指针&#xff1f;异常抛出导致的资源泄漏问题分析 智能指针与RAIIC常用智能指针 使用智能指针优化代码优化后的代码优化点分析 析构函数中的异常问题解决方法 RAII 和智能指针的设计思路详解什么是 RAII&#xff1f;RAII 的…

Android数据存储

前言 在前面&#xff0c;我们已经学了控件和布局&#xff0c;那么我们在存储数据的时候&#xff0c;并不能持久化的存储&#xff0c;所以我们需要来学习一些如何持久化存储数据的方式. 数据存储方式 文件存储&#xff1a;在android中提供了openFileInput()方法和openFileOut…

Java基础——多线程

1. 线程 是一个程序内部的一条执行流程程序中如果只有一条执行流程&#xff0c;那这个程序就是单线程的程序 2. 多线程 指从软硬件上实现的多条执行流程的技术&#xff08;多条线程由CPU负责调度执行&#xff09; 2.1. 如何创建多条线程 Java通过java.lang.Thread类的对象…

【网络】网络层——IP协议

> 作者&#xff1a;დ旧言~ > 座右铭&#xff1a;松树千年终是朽&#xff0c;槿花一日自为荣。 > 目标&#xff1a;了解在网络层下的IP协议。 > 毒鸡汤&#xff1a;有些事情&#xff0c;总是不明白&#xff0c;所以我不会坚持。早安! > 专栏选自&#xff1a;网络…

获取当前程序运行时的栈大小[C语言]

废话前言 一晃已经毕业了4年&#xff0c;也在某个时间点&#xff0c;从面试者转变成了面试官。 进行第一次面试的时候&#xff0c;我好像比候选人还慌张&#xff0c;压根不知道问什么&#xff0c;好在是同行业&#xff0c;看着简历问了一些协议内容以及模块设计思路&#xff0…

人工智能之数学基础:数学在人工智能领域中的地位

人工智能&#xff08;AI&#xff09;是一种新兴的技术&#xff0c;它的目标是构建能够像人类一样思考、学习、推理和解决问题的智能机器。AI已经成为了许多行业的重要组成部分&#xff0c;包括医疗、金融、交通、教育等。而数学则是AI领域中不可或缺的基础学科。本文将阐述数学…

UE5 第一人称射击项目学习(一)

因为工作需要&#xff0c;需要掌握ue5的操作。 选择了视频资料 UE5游戏制作教程Unreal Engine 5 C作为学习。 第一个目标是跟着视频制作出一款第一人称射击项目。 同时作为入门&#xff0c;这个项目不会涉及到C&#xff0c;而是一个纯蓝图的项目。 项目目标 这个项目将实…

图像分类之花卉识别实验验证

本实验基于37种主流的图像分类算法模型&#xff0c;对64种花卉进行识别。使用包括vgg、resnet、densenet、efficientnet、inception、mobilenet等37种图像分类模型进行实验&#xff0c;评估各种模型对花卉的识别准确度、计算量、参数量&#xff0c;对比不同模型的性能和优缺点。…

Linux基础开发工具使用

目录 1. 软件包管理器yum 1.1 概念介绍 1.2 更换镜像源&#xff08;可选&#xff09; 1.3 工具的搜索/查看/安装/卸载 1.4 优势 2. vim编辑器 2.1 vi和vim 2.2 三种常用模式和操作 2.3 配置vim 3. Linux编译器-gcc/g 4. Linux调试器-gdb 5. make和Makefile 6.…

电脑怎么自动切换IP地址

在现代网络环境中&#xff0c;电脑自动切换IP地址的需求日益增多。无论是出于网络安全、隐私保护&#xff0c;还是为了绕过地域限制&#xff0c;自动切换IP地址都成为了许多用户关注的焦点。本文将详细介绍几种实现电脑自动切换IP地址的方法&#xff0c;以满足不同用户的需求。…

PMBOK® 第六版 控制进度

目录 读后感—PMBOK第六版 目录 制定了明确的计划后&#xff0c;对计划的控制尤为重要。例如&#xff0c;经常提到的“累积效应”&#xff0c;如果某个阶段的评分仅为0.9分&#xff0c;那么五个得分为0.9分的阶段&#xff0c;最终结果可能只是一个0.5分。 特别是在当今这个时…

linux001.在Oracle VM VirtualBox中ubuntu虚拟系统扩容

1.打开终端切换到virtualBox安装目录 2.输入命令扩容 如上终端中的代码解释&#xff1a; D:\Program Files\Oracle\VirtualBox>.\VBoxManage modifyhd D:\ubuntu18.04\Ubuntu18.04\Ubuntu18.04.vdi --resize 40960如上代码说明&#xff1a;D:\Program Files\Oracle\Virtual…

Web导出Excel表格

背景&#xff1a; 1. 后端主导实现 流程&#xff1a;前端调用到导出excel接口 -> 后端返回excel文件流 -> 浏览器会识别并自动下载 场景&#xff1a;大部分场景都有后端来做 2. 前端主导实现 流程&#xff1a;前端获取要导出的数据 -> 常规数据用插件处理成一个e…