【第五节】C++设计模式(创建型模式)-Prototype(原型)模式

目录

一、问题背景

二、 模式选择

三、讨论总结


一、问题背景

        在软件开发中,有时我们需要通过已有对象来创建新对象,而不是从头开始构建。这种需求让我想起了现代制造业中的 3D 打印技术。通过扫描一个现有的物体,3D 打印机可以快速复制出多个完全相同的副本,而无需重新设计和建模。这种“复制已有对象”的能力在编程中也有对应的设计模式,那就是 Prototype 模式

        C++中通过拷贝构造函数实现对象复制,但这一机制存在两个关键挑战:

                浅拷贝陷阱:默认拷贝构造的按位复制可能引发指针悬挂问题

                多态复制难题:基类指针无法直接调用派生类的拷贝构造函数

        这些问题既是面试常见考点,也是系统稳定性的潜在威胁。原型模式通过标准化的复制接口,为这些问题提供了系统化的解决方案。

二、 模式选择

        Prototype 模式的结构非常简单,其核心是一个 `Clone` 接口,用于复制对象。在 C++ 中,`Clone` 方法通常通过拷贝构造函数来实现。以下是 Prototype 模式的典型结构图:

原型模式的核心是Clone抽象接口,其标准实现包含三个关键组件:

        Prototype:定义克隆接口的抽象基类

        ConcretePrototype:实现具体克隆操作的可复制对象

        Client:通过原型对象创建新实例的使用者

实现
        为了帮助初学者更好地理解 Prototype 模式,下面将提供一个完整的 C++ 实现示例。代码分为三个部分:`Prototype.h`、`Prototype.cpp` 和 `main.cpp`。

代码片段 1:Prototype.h

// Prototype.h
#ifndef _PROTOTYPE_H_
#define _PROTOTYPE_H_

// 抽象基类 Prototype
class Prototype {
public:
    virtual ~Prototype() = default; // 虚析构函数,确保派生类对象正确释放资源
    virtual Prototype* Clone() const = 0; // 纯虚函数,用于复制对象
protected:
    Prototype() = default; // 保护构造函数,防止直接实例化
};

// 具体实现类 ConcretePrototype
class ConcretePrototype : public Prototype {
public:
    ConcretePrototype() = default; // 默认构造函数
    ConcretePrototype(const ConcretePrototype& cp); // 拷贝构造函数
    ~ConcretePrototype() override = default; // 析构函数
    Prototype* Clone() const override; // 实现 Clone 方法
};

#endif // ~_PROTOTYPE_H_

代码片段 2:Prototype.cpp

// Prototype.cpp
#include "Prototype.h"
#include <iostream>
using namespace std;

// Prototype 类的实现
Prototype::Prototype() {
    // 基类构造函数,用于初始化基类成员(如果有)
}

Prototype::~Prototype() {
    // 基类析构函数,用于释放基类资源(如果有)
}

Prototype* Prototype::Clone() const {
    return nullptr; // 基类中返回空指针,表示不支持直接复制
}

// ConcretePrototype 类的实现
ConcretePrototype::ConcretePrototype() {
    // 默认构造函数,用于初始化具体类成员(如果有)
}

ConcretePrototype::~ConcretePrototype() {
    // 析构函数,用于释放具体类资源(如果有)
}

// 拷贝构造函数
ConcretePrototype::ConcretePrototype(const ConcretePrototype& cp) {
    cout << "ConcretePrototype copy ..." << endl; // 打印拷贝构造信息
    // 如果需要深拷贝,可以在这里实现
}

// 实现 Clone 方法
Prototype* ConcretePrototype::Clone() const {
    return new ConcretePrototype(*this); // 调用拷贝构造函数,返回新对象
}

代码片段 3:main.cpp

// main.cpp
#include "Prototype.h"
#include <iostream>
using namespace std;

int main(int argc, char* argv[]) {
    // 创建原型对象
    Prototype* p = new ConcretePrototype();

    // 复制对象
    Prototype* p1 = p->Clone();

    // 释放资源
    delete p;
    delete p1;

    return 0;
}

代码说明

实现要点

        深拷贝控制:在拷贝构造函数中实现资源复制逻辑

        多态支持:基类声明虚析构函数确保正确释放资源

        接口隔离:保护构造函数强制使用克隆方式创建对象

        Prototype 模式的结构和实现都非常简单,其核心在于 `Clone` 方法的实现。在 C++ 中,`Clone` 方法通常通过拷贝构造函数来完成对象的复制。为了简化示例,这里没有涉及深拷贝(即对象中包含指针或复合对象的情况),而是直接使用了编译器提供的默认拷贝构造函数(按位拷贝)。

        需要注意的是,这种实现方式仅适用于简单场景。如果对象中包含动态分配的资源(如指针),则需要手动实现深拷贝,以避免潜在的内存问题。
 

三、讨论总结

        Prototype 模式通过复制原型对象来创建新对象,这种方式特别适用于以下场景:
(1)当对象的创建过程比较复杂,且需要频繁创建相似对象时。
(2)当需要避免重复初始化对象的开销时。

        与其他创建型模式(如 Builder 模式和 AbstractFactory 模式)相比,Prototype 模式的独特之处在于:
        Builder 模式:侧重于逐步构建复杂对象,而不是直接返回对象。
        AbstractFactory 模式:用于创建一系列相互依赖的对象。
        Prototype 模式:通过复制自身来创建新对象。

        Prototype 模式的核心思想是将对象创建的责任交给对象本身,而不是外部工厂。这种设计不仅提高了代码的灵活性,还减少了重复代码的编写。


        Prototype 模式是一种简单而强大的设计模式,它通过复制现有对象来创建新对象,避免了重复初始化的开销。在 C++ 中,拷贝构造函数是实现 Prototype 模式的关键,但需要注意浅拷贝和深拷贝的问题。通过合理使用 Prototype 模式,可以显著提高代码的复用性和可维护性。

由此原型模式通过标准化对象复制接口,有效解决了以下问题:

        构造开销:避免重复初始化操作

        多态复制:支持通过基类接口复制派生类对象

        状态克隆:完整复制对象的瞬时状态

在C++实现中需特别注意:

        严格实现深拷贝防止内存问题

        通过虚析构函数保证多态正确性

        考虑使用智能指针管理克隆对象生命周期

        该模式在游戏开发(角色克隆)、文档编辑(版本快照)等领域有广泛应用,是构建高效对象创建系统的利器。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

next.js-学习2

next.js-学习2 1. https://nextjs.org/learn/dashboard-app/getting-started2. 模拟的数据3. 添加样式4. 字体&#xff0c;图片5. 创建布局和页面页面导航 1. https://nextjs.org/learn/dashboard-app/getting-started /app: Contains all the routes, components, and logic …

OpenCV计算摄影学(1)图像修复(Inpainting)的函数inpaint()

操作系统&#xff1a;ubuntu22.04 OpenCV版本&#xff1a;OpenCV4.9 IDE:Visual Studio Code 编程语言&#xff1a;C11 算法描述 使用图像中选定区域的邻域来恢复该选定区域。 cv::inpaint 函数是 OpenCV 中用于图像修复&#xff08;Inpainting&#xff09;的一个重要函数。它…

北京智和信通:全方位智能 OLT、ONU 设备监控运维方案

随着网络技术的不断迭代与发展&#xff0c;OLT作为光纤接入网中的核心设备&#xff0c;负责管理多个ONU&#xff0c;实现数据的传输和分配。其监控与运维的重要性愈发凸显&#xff0c;为了确保网络运行的高效与稳定&#xff0c;选择一套全面且高效的OLT、ONU监控运维方案显得尤…

python-leetcode-搜索二维矩阵 II

240. 搜索二维矩阵 II - 力扣&#xff08;LeetCode&#xff09; class Solution:def searchMatrix(self, matrix: List[List[int]], target: int) -> bool:if not matrix or not matrix[0]:return Falsem, n len(matrix), len(matrix[0])i, j 0, n - 1 # 从右上角开始whi…

推送项目 之 解决冲突

文章目录 为什么会发生冲突&#xff1f;如何解决这些冲突&#xff1f;1. **查看冲突文件**2. **解决二进制文件冲突**3. **解决文本文件冲突**4. **标记冲突已解决**5. **完成合并**6. **推送更改** 注意事项总结 问题&#xff1a;我们在git pusll拉取远程仓库的代码到本地对比…

网页版的俄罗斯方块

1、新建一个txt文件 2、打开后将代码复制进去保存 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>俄…

Docker 部署 Jenkins持续集成(CI)工具

[TOC](Docker 部署 Jenkins持续集成(CI)工具) 前言 Jenkins 是一个流行的开源自动化工具&#xff0c;广泛应用于持续集成&#xff08;CI&#xff09;和持续交付&#xff08;CD&#xff09;的环境中。通过 Docker 部署 Jenkins&#xff0c;可以简化安装和配置过程&#xff0c;并…

LLM+多智能体协作:基于CrewAI与DeepSeek的邮件自动化实践

文章目录 引言理解 Flows&#xff08;工作流&#xff09;与 Crews&#xff08;协作组&#xff09;一、环境准备与工具安装1.1 Python环境搭建1.2 创建并激活虚拟环境1.3 安装核心依赖库&#xff08;crewai、litellm&#xff09; 二、本地DeepSeek R1大模型部署2.1 Ollama框架安…

[SQL] 事务的四大特性(ACID)

&#x1f384;事务的四大特性 以下就是事务的四大特性&#xff0c;简称ACID。 原子性&#x1f4e2;事务时不可分割的最小操作单元&#xff0c;要么全部成功&#xff0c;要么全部失败。一致性&#x1f4e2;事务完成后&#xff0c;必须使所有的数据都保持一致隔离性&#x1f4e2…

AI时代前端开发:自主学习能力的培养

在人工智能&#xff08;AI&#xff09;飞速发展的今天&#xff0c;技术迭代的速度如同脱缰的野马&#xff0c;对所有技术人员&#xff0c;特别是前端开发者&#xff0c;都提出了前所未有的挑战。新的框架、库、工具层出不穷&#xff0c;稍有松懈&#xff0c;就会被时代抛在身后…

【备赛】点亮LED

LED部分的原理图 led前面有锁存器&#xff0c;这是为了防止led会受到lcd的干扰&#xff08;lcd也需要用到这些引脚&#xff09;。 每次想要对led操作&#xff0c;就需要先打开锁存器&#xff0c;再执行操作&#xff0c;最后关闭锁存器。 这里需要注意的是&#xff0c;引脚配置…

Rocky8 源码安装 HAProxy

HAProxy 是一款开源的高性能 负载均衡器 和 反向代理 软件&#xff0c;专注于处理高并发流量分发&#xff0c;广泛应用于企业级架构中提升服务的可用性、扩展性和安全性。 一、HAProxy 简介 1.1.HAProxy 是什么&#xff1f; 本质&#xff1a; 基于 C 语言开发 的轻量级工具&a…

Javascript网页设计案例:通过PDFLib实现一款PDF分割工具,分割方式自定义-完整源代码,开箱即用

功能预览 一、工具简介 PDF 分割工具支持以下核心功能: 拖放或上传 PDF 文件:用户可以通过拖放或点击上传 PDF 文件。两种分割模式: 指定范围:用户可以指定起始页和结束页,提取特定范围的内容。固定间距:用户可以设置间隔页数(例如每 5 页分割一次),工具会自动完成分…

微信小程序调用火山方舟(字节跳动火山引擎)中的DeepSeek大模型

一、注册火山引擎账号&#xff0c;创建API Key和model&#xff08;接入点ID&#xff09; 1.注册并登陆火山引擎账号&#xff0c;网址为&#xff1a;https://console.volcengine.com/ 2.根据登陆后的页面提示进行实名认证&#xff0c;实名认证后才能创建API Keyt和创建接入点。…

Spring Boot 应用(官网文档解读)

Spring Boot 启动方式 SpringApplication.run(MyApplication.class, args); Spring Boot 故障分析器 在Spring Boot 项目启动发生错误的时候&#xff0c;我们通常可以看到上面的内容&#xff0c;即 APPLICATION FAILED TO START&#xff0c;以及后面的错误描述。这个功能是通过…

从单片机的启动说起一个单片机到点灯发生了什么下——使用GPIO点一个灯

目录 前言 HAL库对GPIO的抽象 核心分析&#xff1a;HAL_GPIO_Init 前言 我们终于到达了熟悉的地方&#xff0c;对GPIO的初始化。经过漫长的铺垫&#xff0c;我们终于历经千辛万苦&#xff0c;来到了这里。关于GPIO的八种模式等更加详细的细节&#xff0c;由于只是点个灯&am…

【JavaWeb学习Day19】

Tlias智能学习系统&#xff08;员工管理&#xff09; 删除员工&#xff1a; 需求分析&#xff1a; 其实&#xff0c;删除单条数据也是一种特殊的批量删除&#xff0c;所以&#xff0c;删除员工的功能&#xff0c;我们只需要开发一个接口就行了。 三层架构&#xff1a; Cont…

Windows 上源码安装 FastGPT

FastGPT 是一个强大的 AI RAG 平台&#xff0c;值得我们去学习了解。与常见的 Python 体系不同&#xff0c;Fast GPT 采用 Node.js/Next.js 平台&#xff08;对于广大 JS 开发者或前端开发者比较亲切友好&#xff09;&#xff0c;安装或部署比较简单。虽然一般情况下推荐简单的…

FreiHAND (handposeX-json 格式)数据集-release >> DataBall

FreiHAND &#xff08;handposeX-json 格式&#xff09;数据集-release 注意&#xff1a; 1)为了方便使用&#xff0c;按照 handposeX json 自定义格式存储 2)使用常见依赖库进行调用,降低数据集使用难度。 3)部分数据集获取请加入&#xff1a;DataBall-X数据球(free) 4)完…

Sinusoidal、RoPE和可学习嵌入的详细介绍及它们增强位置感知能力的示例

前文,我们已经构建了一个小型的字符级语言模型,是在transformer架构基础上实现的最基本的模型,我们肯定是希望对该模型进行改进和完善的。所以我们的另外一篇文章也从数据预处理、模型架构、训练策略、评估方法、代码结构、错误处理、性能优化等多个方面提出具体的改进点,但…