C++实现设计模式---原型模式 (Prototype)

news/2025/1/15 10:31:13/

原型模式 (Prototype)

原型模式 是一种创建型设计模式,它通过复制现有对象来创建新对象,而不是通过实例化。


意图

  • 使用原型实例指定要创建的对象类型,并通过复制该原型来生成新对象。
  • 提供一种高效创建对象的方式,尤其是当对象的创建成本较高时。

使用场景

  1. 对象创建成本高
    • 如果直接实例化对象开销较大(如对象初始化需要大量资源),可以通过克隆已存在的对象快速生成新实例。
  2. 复杂对象需要重复创建
    • 当对象的结构较复杂,需要创建多个相似对象时。
  3. 避免直接依赖具体类
    • 原型模式通过复制实例而不是直接依赖构造函数,可以减少对具体类的依赖。

参与者角色

  1. 原型接口 (Prototype)
    • 定义一个克隆方法,用于复制对象。
  2. 具体原型 (Concrete Prototype)
    • 实现原型接口,定义克隆方法,返回当前实例的复制品。
  3. 客户端 (Client)
    • 使用原型接口创建新对象,而不直接依赖具体类。

示例代码

以下示例展示了如何使用原型模式创建几种不同类型的图形对象。

#include <iostream>
#include <memory>
#include <string>// 原型接口
class Shape {
public:virtual ~Shape() {}virtual std::unique_ptr<Shape> clone() const = 0;virtual void draw() const = 0;
};// 具体原型:圆形
class Circle : public Shape {
private:int radius;public:Circle(int r) : radius(r) {}// 实现克隆方法std::unique_ptr<Shape> clone() const override {return std::make_unique<Circle>(*this);}void draw() const override {std::cout << "Drawing a Circle with radius: " << radius << std::endl;}
};// 具体原型:矩形
class Rectangle : public Shape {
private:int width, height;public:Rectangle(int w, int h) : width(w), height(h) {}// 实现克隆方法std::unique_ptr<Shape> clone() const override {return std::make_unique<Rectangle>(*this);}void draw() const override {std::cout << "Drawing a Rectangle with width: " << width<< " and height: " << height << std::endl;}
};// 客户端代码
int main() {// 创建具体原型std::unique_ptr<Shape> circlePrototype = std::make_unique<Circle>(10);std::unique_ptr<Shape> rectanglePrototype = std::make_unique<Rectangle>(20, 30);// 克隆原型auto circle1 = circlePrototype->clone();auto circle2 = circlePrototype->clone();auto rectangle1 = rectanglePrototype->clone();// 使用克隆对象circle1->draw(); // 输出:Drawing a Circle with radius: 10circle2->draw(); // 输出:Drawing a Circle with radius: 10rectangle1->draw(); // 输出:Drawing a Rectangle with width: 20 and height: 30return 0;
}

代码解析

1. 原型接口

  • 定义了 clone() 方法,用于创建当前对象的副本:
virtual std::unique_ptr<Shape> clone() const = 0;

2. 具体原型类

  • CircleRectangle 是具体原型类,分别实现了 clone() 方法,返回自身的副本:
std::unique_ptr<Shape> clone() const override {return std::make_unique<Circle>(*this);
}

3. 客户端

  • 客户端通过调用 clone() 方法创建对象,而无需直接依赖具体类:
auto circle1 = circlePrototype->clone();
circle1->draw();

优缺点

优点

  1. 对象复制高效
    • 直接复制对象比重新实例化速度更快。
  2. 减少依赖
    • 客户端代码无需依赖具体类的构造函数。
  3. 动态创建对象
    • 可在运行时动态生成对象,而不是在编译时确定。

缺点

  1. 深拷贝复杂性
    • 如果对象中包含指针或其他复杂资源,需特别注意深拷贝逻辑。
  2. 实现复杂
    • 需要为每个类实现 clone() 方法,增加了一定的代码量。

适用场景

  • 需要大量相似对象:如图形编辑器中需要重复创建类似图形。
  • 对象初始化成本高:如大型游戏中加载模型、地图等资源。
  • 需要动态生成对象:如根据配置文件动态生成对象。

改进与扩展

  1. 深拷贝支持

    • 如果对象包含动态内存或指针成员变量,需要实现深拷贝以避免资源共享问题。
  2. 结合原型注册表

    • 使用注册表保存所有原型实例,根据名称或类型动态克隆对象:
    class PrototypeRegistry {
    private:std::unordered_map<std::string, std::unique_ptr<Shape>> prototypes;public:void registerPrototype(const std::string& name, std::unique_ptr<Shape> prototype) {prototypes[name] = std::move(prototype);}std::unique_ptr<Shape> create(const std::string& name) const {return prototypes.at(name)->clone();}
    };
    

总结

原型模式通过复制已有对象来快速创建新对象,避免了复杂的初始化逻辑,并减少了对具体类的依赖。
它适用于需要动态生成大量相似对象或高效创建对象的场景。


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

相关文章

初阶数据结构【双链表及其接口的实现】

目录 前言一、基本结构二、双链表的接口实现2.1 双链表基本功能接口2.1.1 双向链表打印2.1.2 申请一个节点2.1.3 创建并返回双向链表的头结点2.1.4 双向链表清理&#xff08;不销毁&#xff09;2.1.5 双向链表销毁 2.2 双向链表增加节点接口2.2.1 双向链表头插2.2.2 双向链表尾…

如何选择 Redis 持久化方式?RDB 和 AOF 的优缺点分析

前言 在开发中&#xff0c;Redis 通常作为缓存使用。由于其特性&#xff0c;查询数据通常比直接访问关系型数据库快得多。但如果未开启持久化&#xff0c;内存中的缓存可能会因意外丢失&#xff0c;尤其是还未同步到持久层的数据。这让持久化变得非常重要。本文将分享我对 Redi…

jupyter notebook练手项目:线性回归——学习时间与成绩的关系

线性回归——学习时间与学习成绩的关系 第1步&#xff1a;导入工具库 pandas——数据分析库&#xff0c;提供了数据结构&#xff08;如DataFrame和Series&#xff09;和数据操作方法&#xff0c;方便对数据集进行读取、清洗、转换等操作。 matplotlib——绘图库&#xff0c;p…

脚本化挂在物理盘、nfs、yum、pg数据库、nginx(已上传脚本)

文章目录 前言一、什么是脚本化安装二、使用步骤1.物理磁盘脚本挂载&#xff08;离线&#xff09;2.yum脚本化安装&#xff08;离线&#xff09;3.nfs脚本化安装&#xff08;离线&#xff09;4.pg数据库脚本化安装&#xff08;离线&#xff09;5.nginx脚本化安装&#xff08;离…

第六届土木建筑及灾害防控国际学术会议暨第三届智慧城市建筑与基础设施耐久性国际学术会议(CADPC DuraBI 2025)

第六届土木建筑及灾害防控国际学术会议暨第三届智慧城市建筑与基础设施耐久性国际学术会议&#xff08;CADPC & DuraBI 2025&#xff09;将于2025年2月28日-3月2日在青岛举办。会议将以“建筑技术”、“灾害预测”、“灾害防控”、“灾后重建”等主题展开学术研讨&#xff…

Java IDEA中Gutter Icons图标的含义

前些天发现了一个蛮有意思的人工智能学习网站,8个字形容一下"通俗易懂&#xff0c;风趣幽默"&#xff0c;感觉非常有意思,忍不住分享一下给大家。 &#x1f449;点击跳转到教程 前言&#xff1a; 很多人刚开始用IDEA来学习编程&#xff0c;会发现下面这些图标。 但是…

《视听导报》是什么类型的报纸?报纸上发文章要交版面费吗?

作为个人成果发表的重要场所&#xff0c;报纸目前正得到越来越多单位的认可。不过在投稿时&#xff0c;我们既要考虑投稿的报纸是否符合评审标准&#xff0c;也要考虑发表文章的成本是否在我们的承受范围之内。 下面就让我们以《视听导报》为例&#xff0c;了解下如何查看报纸的…

【LeetCode: 240. 搜索二维矩阵 II + 指针 + 遍历】

&#x1f680; 算法题 &#x1f680; &#x1f332; 算法刷题专栏 | 面试必备算法 | 面试高频算法 &#x1f340; &#x1f332; 越难的东西,越要努力坚持&#xff0c;因为它具有很高的价值&#xff0c;算法就是这样✨ &#x1f332; 作者简介&#xff1a;硕风和炜&#xff0c;…