c++ 原型模式

2024-09-02 07:04
文章标签 c++ 模式 原型

本文主要是介绍c++ 原型模式,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

    • 什么是原型模式
    • 为什么要使用原型模式
    • 使用场景
    • 示例

什么是原型模式

用原型实例指定创建对象的种类,并通过拷贝这些原型创建新的对象,简单理解就是“克隆指定对象

为什么要使用原型模式

原型模式(Prototype Pattern)是一种创建型设计模式,它允许通过复制现有对象来创建新对象,而不是通过标准的构造函数来创建。在C++中使用原型模式有以下几个主要原因:

  • 避免重复的初始化代码:
    如果一个对象的初始化过程非常复杂且耗时,使用原型模式可以避免每次创建新对象时都重复执行这些初始化步骤。通过复制一个已经初始化好的对象,可以节省大量的时间和资源。
  • 动态创建对象:
    在某些情况下,你需要在运行时决定创建哪种类型的对象。原型模式允许你预先创建一些原型对象,然后在需要时通过复制这些原型来创建新对象,这样可以灵活地动态创建对象。
  • 隐藏对象创建的复杂性:
    原型模式可以将对象的创建过程封装起来,客户端代码不需要知道具体的创建细节。这有助于降低系统的耦合度,使得代码更加模块化和易于维护。
  • 支持对象的深拷贝:
    在C++中,默认的拷贝构造函数和赋值操作符只能进行浅拷贝。如果对象内部包含指针或其他资源,浅拷贝可能导致资源共享和潜在的内存问题。原型模式可以实现深拷贝,确保每个新对象都是独立的,避免资源共享的问题。

在这里插入图片描述

使用场景

1. 游戏开发在游戏中,经常需要创建大量的游戏对象,如敌人、道具等。如果每次创建这些对象都重新初始化,可能会导致性能问题。通过使用原型模式,可以先创建一个对象作为原型,然后在需要的时候通过克隆这个原型对象来生成新的实例,这样可以节省初始化的时间。
2. 报表系统报表系统中往往需要生成大量的类似数据表格。如果每个表格都需要从头开始创建,则会消耗很多资源。通过使用原型模式,可以先创建一个基本的表格模板,然后根据需要克隆这个模板并填充不同的数据,从而快速生成多个表格实例。
3. GUI 应用在图形用户界面(GUI)应用中,经常需要创建许多具有相同外观但内容不同的组件(如按钮、列表项)。使用原型模式可以让开发者仅需创建一个基础组件作为原型,之后可以通过克隆并修改这个原型来创建新的组件实例。
4. 数据库操作在数据库操作中,有时候需要批量插入数据或者更新记录。如果每次操作都需要重新构建对象,则效率低下。可以先构建一个数据对象作为原型,然后通过克隆这个对象并修改特定字段值来创建新的数据对象,从而优化数据库操作。
5. 测试环境在软件测试中,尤其是单元测试和集成测试时,经常需要构造测试数据。如果直接创建测试数据对象比较耗时,可以采用原型模式来预先创建一个或几个完整的测试数据对象作为原型,然后通过复制这些原型来快速生成所需的测试数据。
6. 内存受限环境在某些内存受限的环境中,比如嵌入式系统,频繁地创建和销毁对象会带来较大的开销。通过原型模式可以有效地减少内存分配和回收的操作,从而优化系统的性能。

示例

#include <iostream>
#include <vector>// 抽象基类 Shape
class Shape {
public:virtual ~Shape() {}virtual Shape* clone() const = 0;virtual void draw() const = 0;
};// Rectangle 类
class Rectangle : public Shape {int width, height;
public:Rectangle(int w, int h) : width(w), height(h) {}Shape* clone() const {return new Rectangle(*this); // 深拷贝}void draw() const {std::cout << "Drawing Rectangle with width: " << width << " and height: " << height << std::endl;}
};// Circle 类
class Circle : public Shape {int radius;
public:Circle(int r) : radius(r) {}Shape* clone() const {return new Circle(*this); // 深拷贝}void draw() const {std::cout << "Drawing Circle with radius: " << radius << std::endl;}
};// 测试函数
void testPrototypePattern() {Shape* rect = new Rectangle(5, 10);Shape* circle = new Circle(7);// 克隆并绘制Shape* clonedRect = rect->clone();Shape* clonedCircle = circle->clone();clonedRect->draw();clonedCircle->draw();// 释放内存delete rect;delete circle;delete clonedRect;delete clonedCircle;
}int main() {testPrototypePattern();return 0;
}

优点

  • 如果创建新的对象比较复杂,可以利用原型模式简化对象的创建过程,同时也能够提高效率。
  • 简化对象的创建,无需理会创建过程。
  • 可以在程序运行时(对象属性发生了变化)获得一份内容相同的实例,他们之间不会相互干扰

缺点

  • 每一个类都必须配备一个克隆方法,对于已有的没有克隆方法的类来说是致命的。

这篇关于c++ 原型模式的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1129320

相关文章

【C++ Primer Plus习题】13.4

大家好,这里是国中之林! ❥前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站。有兴趣的可以点点进去看看← 问题: 解答: main.cpp #include <iostream>#include "port.h"int main() {Port p1;Port p2("Abc", "Bcc", 30);std::cout <<

C++包装器

包装器 在 C++ 中,“包装器”通常指的是一种设计模式或编程技巧,用于封装其他代码或对象,使其更易于使用、管理或扩展。包装器的概念在编程中非常普遍,可以用于函数、类、库等多个方面。下面是几个常见的 “包装器” 类型: 1. 函数包装器 函数包装器用于封装一个或多个函数,使其接口更统一或更便于调用。例如,std::function 是一个通用的函数包装器,它可以存储任意可调用对象(函数、函数

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

06 C++Lambda表达式

lambda表达式的定义 没有显式模版形参的lambda表达式 [捕获] 前属性 (形参列表) 说明符 异常 后属性 尾随类型 约束 {函数体} 有显式模版形参的lambda表达式 [捕获] <模版形参> 模版约束 前属性 (形参列表) 说明符 异常 后属性 尾随类型 约束 {函数体} 含义 捕获:包含零个或者多个捕获符的逗号分隔列表 模板形参:用于泛型lambda提供个模板形参的名

在JS中的设计模式的单例模式、策略模式、代理模式、原型模式浅讲

1. 单例模式(Singleton Pattern) 确保一个类只有一个实例,并提供一个全局访问点。 示例代码: class Singleton {constructor() {if (Singleton.instance) {return Singleton.instance;}Singleton.instance = this;this.data = [];}addData(value)

6.1.数据结构-c/c++堆详解下篇(堆排序,TopK问题)

上篇:6.1.数据结构-c/c++模拟实现堆上篇(向下,上调整算法,建堆,增删数据)-CSDN博客 本章重点 1.使用堆来完成堆排序 2.使用堆解决TopK问题 目录 一.堆排序 1.1 思路 1.2 代码 1.3 简单测试 二.TopK问题 2.1 思路(求最小): 2.2 C语言代码(手写堆) 2.3 C++代码(使用优先级队列 priority_queue)

【C++高阶】C++类型转换全攻略:深入理解并高效应用

📝个人主页🌹:Eternity._ ⏩收录专栏⏪:C++ “ 登神长阶 ” 🤡往期回顾🤡:C++ 智能指针 🌹🌹期待您的关注 🌹🌹 ❀C++的类型转换 📒1. C语言中的类型转换📚2. C++强制类型转换⛰️static_cast🌞reinterpret_cast⭐const_cast🍁dynamic_cast 📜3. C++强制类型转换的原因📝

C++——stack、queue的实现及deque的介绍

目录 1.stack与queue的实现 1.1stack的实现  1.2 queue的实现 2.重温vector、list、stack、queue的介绍 2.1 STL标准库中stack和queue的底层结构  3.deque的简单介绍 3.1为什么选择deque作为stack和queue的底层默认容器  3.2 STL中对stack与queue的模拟实现 ①stack模拟实现

c++的初始化列表与const成员

初始化列表与const成员 const成员 使用const修饰的类、结构、联合的成员变量,在类对象创建完成前一定要初始化。 不能在构造函数中初始化const成员,因为执行构造函数时,类对象已经创建完成,只有类对象创建完成才能调用成员函数,构造函数虽然特殊但也是成员函数。 在定义const成员时进行初始化,该语法只有在C11语法标准下才支持。 初始化列表 在构造函数小括号后面,主要用于给