“对象创建”模式之原型模式

目录

  • Prototype 原型模式
    • 动机 Motivation
    • 引例
    • 模式定义
    • 结构 Structure
    • 要点总结

Prototype 原型模式

动机 Motivation

  • 在软件系统中,经常面临着“某些结构复杂的对象”的创建工作;由于需求的变化,这些对象经常面临着剧烈的变化,但是它们却拥有比较稳定一致的接口。
  • 如何应对这种变化?如何向“客户程序(使用这些对象的程序)”隔离出“这些易变对象”,从而使得“依赖这些易变对象的客户程序”不随着需求改变而改变?

引例

  • 前文中分割器的例子,使用工厂方法模式是一种方式,下面介绍使用原型模式的方式。
// 抽象类
class ISplitter
{
public:virtual void split() = 0;virtual ISplitter* Clone() = 0;          // 通过克隆自己来创建对象virtual ~ISplitter() {}
};// 具体类 - 二进制分割器
class BinarySplitter : public ISplitter
{
private:std::string* binaryData;  // 假设有需要深拷贝的指针成员int dataSize;public:BinarySplitter() : binaryData(nullptr), dataSize(0) {}// 深拷贝构造函数BinarySplitter(const BinarySplitter& other) : dataSize(other.dataSize){if (other.binaryData != nullptr) {binaryData = new std::string(*other.binaryData);} else {binaryData = nullptr;}}~BinarySplitter() {delete binaryData;}void setData(const std::string& data) {if (binaryData == nullptr) {binaryData = new std::string(data);} else {*binaryData = data;}}virtual void split() override {std::cout << "Splitting binary data: " << (binaryData ? *binaryData : "") << std::endl;}virtual ISplitter* Clone() override {return new BinarySplitter(*this);  // 调用拷贝构造函数进行深拷贝}
};// 具体类 - 图片分割器
class PictureSplitter : public ISplitter
{
private:std::string* imageData;  // 假设有需要深拷贝的指针成员int width, height;public:PictureSplitter() : imageData(nullptr), width(0), height(0) {}// 深拷贝构造函数PictureSplitter(const PictureSplitter& other) : width(other.width), height(other.height){if (other.imageData != nullptr) {imageData = new std::string(*other.imageData);} else {imageData = nullptr;}}~PictureSplitter() {delete imageData;}void setImage(const std::string& data, int w, int h) {if (imageData == nullptr) {imageData = new std::string(data);} else {*imageData = data;}width = w;height = h;}virtual void split() override {std::cout << "Splitting picture (" << width << "x" << height << "): " << (imageData ? imageData->substr(0, 10) + "..." : "") << std::endl;}virtual ISplitter* Clone() override {return new PictureSplitter(*this);  // 调用拷贝构造函数进行深拷贝}
};// 具体类 - 视频分割器
class VideoSplitter : public ISplitter
{
private:std::string* videoData;  // 假设有需要深拷贝的指针成员double duration;public:VideoSplitter() : videoData(nullptr), duration(0) {}// 深拷贝构造函数VideoSplitter(const VideoSplitter& other) : duration(other.duration){if (other.videoData != nullptr) {videoData = new std::string(*other.videoData);} else {videoData = nullptr;}}~VideoSplitter() {delete videoData;}void setVideo(const std::string& data, double dur) {if (videoData == nullptr) {videoData = new std::string(data);} else {*videoData = data;}duration = dur;}virtual void split() override {std::cout << "Splitting video (" << duration << "s): " << (videoData ? videoData->substr(0, 10) + "..." : "") << std::endl;}virtual ISplitter* Clone() override {return new VideoSplitter(*this);  // 调用拷贝构造函数进行深拷贝}
};// MainForm
class MainForm
{ISplitter* prototype;public:MainForm(ISplitter* prototype)          // 通常由外部传入{this->prototype = prototype;}~MainForm() {delete prototype;}void Button1_Click(){ISplitter* splitter = prototype->Clone();     // 克隆原型    splitter->split();delete splitter;}
};int main() 
{// 创建原型对象BinarySplitter* binaryProto = new BinarySplitter();binaryProto->setData("Sample binary data");// 使用原型MainForm form1(binaryProto);form1.Button1_Click();// 另一个例子PictureSplitter* pictureProto = new PictureSplitter();pictureProto->setImage("Very long picture data...", 1920, 1080);MainForm form2(pictureProto);form2.Button1_Click();return 0;
}

模式定义

使用原型实例指定创建对象的种类,然后通过拷贝这些原型来创建新的对象。

结构 Structure

在这里插入图片描述
前文代码中的ISplitter对应图中的Prototype;BinarySplitter对应ConcretePrototype1,VideoSplitter对应ConcretePrototype2;MainForm对应Client

要点总结

  • Prototype原型模式同样用于隔离类对象的使用者和具体类型(易变类)之间的耦合关系,它同样要求这些“易变类”拥有“稳定的接口”。
  • 原型模式对于“如何创建易变类的实体对象”采用“原型克隆”的方法来做,它使得我们可以非常灵活地动态创建“拥有某些稳定接口”的新对象——所需工作仅仅是注册一个新类的对象(即原型),然后在任何需要的地方Clone。
  • 原型模式中的Clone方法可以利用某些框架中的序列化来实现深拷贝。

来源:极客班——C++设计模式入门

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Tomcat服务概述

前言&#xff1a; 作为Apache软件基金会Jakarta项目的核心成果&#xff0c;Tomcat凭借其轻量级、开源免费的特性&#xff0c;已成为Java Web应用服务的行业基准。它实现了完整的Servlet与JSP规范&#xff0c;通过模块化架构&#xff08;Connector请求处理层与Container业务逻辑…

HarmonyOS应用开发高级认证知识点梳理 (一) 布局与样式

以下是 HarmonyOS 应用开发中 ‌布局与样式‌ 的核心知识点梳理&#xff08;针对高级认证备考&#xff09;&#xff0c;结合官方文档与高频考点&#xff1a; 一、布局系统核心知识点 布局容器类型‌ 线性布局‌&#xff1a;Column&#xff08;纵向&#xff09;、Row&#xf…

【Ragflow】30.离线环境迁移方案

前言 之前的 Ragflow-Plus 在服务器上稳定运行一段时间后&#xff0c;接到任务&#xff0c;要把服务迁移到一台古老的&#xff0c;离线的windows台式机上。 起初认为&#xff0c;下载离线安装包&#xff0c;加载离线镜像&#xff0c;迁移下数据就可以了。 结果坑多得意想不到…

nrf52840蓝牙学习(定时器的应用)

和其他 MCU 处理器一样&#xff0c;在 nrf52840 中定时器的功能是十分强大的。其内部包含了 5 个定时 器 TIMER 模块 :TIMER0 、 TIMER1 、 TIMER2 、 TIMER3 、 TIMER4 &#xff0c;如下表 10.1 所示。 1. 时钟源 首先定时器 TIMER 工作在高频时钟源&#xff08…

【Bluedroid】蓝牙启动之BTM_reset_complete源码解析

当蓝牙控制器完成硬件重置后,协议栈需通过一系列初始化操作恢复各模块状态。本文深入分析BTM_reset_complete核心函数及其调用链,详解 L2CAP 连接清理、安全模块重置、扫描参数恢复、BLE 隐私功能初始化等关键流程,揭示蓝牙设备在重置后如何通过标准化状态恢复确保互操作性、…

containerd 项目主要目录简要说明

containerd 项目结构清晰&#xff0c;核心代码分布在若干主目录下。以下是 client、cmd、core、internal、pkg、plugins 这六个主要包/目录的简要作用说明&#xff1a; 1. client 作用&#xff1a;封装与 containerd 守护进程通信的 Go 客户端 API&#xff0c;主要基于 gRPC。…

有线转无线工具,轻松创建WiFi热点

软件介绍 今天为大家推荐一款实用的无线网络共享工具——MyPublicWiFi。这款软件能够将电脑的有线网络转换为无线WiFi&#xff0c;方便其他设备连接使用。 安装与设置 该软件为安装版程序&#xff0c;安装完成后会自动识别当前电脑的IP地址。用户可在软件界面中自定义设…

Linux下,通过标准I2C驱动读取Sensor ID

sensor型号&#xff1a;OS04L10&#xff0c;sensor引脚以及时钟要先配置好&#xff0c;源码如下&#xff1a; #include <fcntl.h> #include <linux/i2c-dev.h> #include <linux/i2c.h> #include <stdint.h> #include <stdio.h> #include <sy…

人工智能基石:SVM支持向量机全解析(附Python实战)

大家好&#xff01;今天我们来深入探讨支持向量机&#xff08;Support Vector Machine, SVM&#xff09;——这个在​​图像识别、文本分类​​等领域广泛应用的强大算法。既能处理分类问题&#xff0c;又能解决回归任务&#xff0c;甚至在非线性数据面前也能游刃有余。本文将带…

mysql查看数据库

在 MySQL 中查看当前数据库的创建语句&#xff0c;使用 SHOW CREATE DATABASE 命令&#xff0c;以下是详细操作指南&#xff1a; 1. 查看当前数据库的创建语句 SHOW CREATE DATABASE database_name; 替换 database_name 为你的数据库名使用反引号 包裹特殊名称或保留字 2.…

ArrayList剖析

大家天天在用List&#xff0c;ArrayList一般来讲应该是程序员用的最多的集合类了。 我们今天研究一下ArrayList。 总体来讲&#xff0c;从底层数据结构或者源码的角度看&#xff0c;List比Map或者Set要简单。 底层数据结构 ArryList其实就是可变长数组。 初始化的时候&…

回顾JAVA中的锁机制

Java中的锁机制 在Java中&#xff0c;锁机制是多线程编程里保障数据一致性与线程安全的关键技术。 1. 内置锁&#xff1a;synchronized关键字 synchronized是Java的内置锁机制&#xff0c;能够保证在同一时刻&#xff0c;只有一个线程可以执行被其修饰的代码块或方法。 用法…

YOLOv11: AN OVERVIEW OF THE KEY ARCHITECTURAL ENHANCEMENTS目标检测论文精读(逐段解析)

YOLOv11: AN OVERVIEW OF THE KEY ARCHITECTURAL ENHANCEMENTS目标检测论文精读&#xff08;逐段解析&#xff09; 论文地址&#xff1a;https://www.arxiv.org/abs/2410.17725 Rahima Khanam and Muhammad Hussain Ultralytics公司发布 CVPR 2024 论文写的比较简单&#xff…

【Erdas实验教程】025:遥感图像辐射增强(雾霾去除)

文章目录 一、雾霾去除原理二、雾霾去除案例一、雾霾去除原理 遥感影像雾霾去除的核心原理是消除大气散射对电磁波的干扰,恢复地物真实反射信息。Haze Reduction 工具的原理: 该工具基于暗目标法(Dark Object Subtraction, DOS),适用于去除因大气散射(雾霾本质是大气颗…

Language Models are Unsupervised Multitask Learners :语言模型是无监督的多任务学习者

摘要 自然语言处理任务&#xff0c;如问答、机器翻译、阅读理解和摘要&#xff0c;通常通过在特定任务的数据集上进行监督学习来解决。我们展示了语言模型在训练于一个包含数百万网页的新数据集——WebText——时&#xff0c;可以无需任何显式监督就开始学习这些任务。当模型以…

SQL语句全攻略:从基础到进阶的编程之旅

目录 一、引言二、SQL 基础语法2.1 SQL 语句写法顺序2.2 关联查询2.3 数据处理常用函数和运算符 三、数据库和表的基本操作3.1 创建数据库3.2 使用数据库3.3 创建表 四、基础增删改查操作4.1 插入数据&#xff08;增&#xff09;4.2 查询数据&#xff08;查&#xff09;4.3 更新…

Kafka的下载安装

目录 一、前期准备 1、查看网卡&#xff1a; 2、配置静态IP 3、设置主机名 4、配置IP与主机名映射 5、关闭防火墙 6、配置免密登录 二、JDK的安装 三、Zookeeper的安装 四、Kafka的安装 1、Kafka的下载安装 2、修改配置文件 4、分发文件 5、修改其他节点broker.i…

opencv入门(6) TrackBar调整图片和键盘响应

文章目录 1 创建trackbar2 使用userdata传入函数3 键盘响应 1 创建trackbar 1.trackbar名称 2.创建在哪个窗口上 3.拖动trackbar改变的值 4.trackBar的最大值 5.trackbar改变时的回调函数 6. 带入回调函数的数据&#xff0c;可以不用带,是一个void指针 createTrackbar(“Value …

QT<33> 修改窗口标题栏背景颜色

前言&#xff1a; 在做项目或者开发QT软件时&#xff0c;如果想要修改窗口标题栏背景颜色&#xff0c;发现没有代码可以直接设置&#xff0c;目前有两种方法可以设置。 第一种&#xff0c;自定义一个界面类&#xff0c;用QLabelQWidget实现&#xff0c;QLabel当作标题栏。 第…

JavaEE-博客系统项目

项目介绍 准备工作 创建数据表 创建项目 添加依赖 创建对应目录 除了基本的数据层mapper&#xff0c;业务层service&#xff0c;交互层controller&#xff0c;还创建了公共类的层common&#xff0c;枚举类层enums&#xff0c;异常类层&#xff0c;和实体类层pojo。 配置项目配…