【设计模式--原型模式(Prototype Pattern)
一、什么是原型模式
原型模式(Prototype Pattern)是一种创建型设计模式,它的主要目的是通过复制现有对象来创建新的对象,而无需显式地使用构造函数或工厂方法。这种模式允许我们创建一个可定制的原型对象,然后通过复制它来创建新的对象,从而避免了重复构建相似的对象。
在原型模式中,通常有以下几个核心角色:
- 原型(Prototype):定义了一个克隆自己的接口,它是需要复制的对象的抽象表示。
- 具体原型(Concrete Prototype):实现了原型接口,实现了克隆自己的方法。
- 客户端(Client):负责创建新对象,通过克隆已有的原型来获得新对象的副本。
原型模式的优点包括:
- 减少了对象的创建成本:通过克隆已有对象,避免了重新构建相似对象的成本。
- 提高了性能:克隆比创建对象更高效,特别是在初始化成本高的情况下。
- 隐藏了对象的创建细节:客户端不需要知道如何创建对象,只需通过克隆获得所需的对象。
然而,原型模式也有一些缺点:
- 如果对象的构造函数有副作用,那么克隆可能会导致不一致的行为。
- 如果对象层次较深,需要逐层克隆,可能会变得比较复杂。
- 在某些语言或环境中,克隆对象可能需要编写较多的代码。
总之,原型模式适用于需要创建相似对象或对象层次的场景,它提供了一种更高效、灵活的对象创建方式。
二、原型模式的代码样例
以下是一个使用C++实现原型模式的简单示例:
#include <iostream>
#include <string>
#include <unordered_map>// 原型抽象类
class Prototype {
public:virtual Prototype* clone() const = 0;virtual void display() const = 0;
};// 具体原型类A
class ConcretePrototypeA : public Prototype {
public:ConcretePrototypeA(int value) : value(value) {}Prototype* clone() const override {return new ConcretePrototypeA(value);}void display() const override {std::cout << "ConcretePrototypeA with value: " << value << std::endl;}private:int value;
};// 具体原型类B
class ConcretePrototypeB : public Prototype {
public:ConcretePrototypeB(const std::string& name) : name(name) {}Prototype* clone() const override {return new ConcretePrototypeB(name);}void display() const override {std::cout << "ConcretePrototypeB with name: " << name << std::endl;}private:std::string name;
};// 原型管理器
class PrototypeManager {
public:void registerPrototype(const std::string& key, Prototype* prototype) {prototypes[key] = prototype;}Prototype* getPrototype(const std::string& key) {if (prototypes.find(key) != prototypes.end()) {return prototypes[key]->clone();}return nullptr;}private:std::unordered_map<std::string, Prototype*> prototypes;
};int main() {PrototypeManager manager;manager.registerPrototype("prototypeA", new ConcretePrototypeA(100));manager.registerPrototype("prototypeB", new ConcretePrototypeB("PrototypeB"));Prototype* clonedA = manager.getPrototype("prototypeA");if (clonedA) {clonedA->display();delete clonedA;}Prototype* clonedB = manager.getPrototype("prototypeB");if (clonedB) {clonedB->display();delete clonedB;}return 0;
}
在这个示例中,我们定义了一个原型抽象类 Prototype,然后有两个具体的原型类 ConcretePrototypeA 和 ConcretePrototypeB,它们都实现了克隆自己的方法。原型管理器 PrototypeManager 用于管理不同类型的原型对象。
通过原型模式,我们可以注册不同的原型对象,并在需要时通过原型管理器创建它们的克隆,从而实现了对象的复制和创建。这种方式可以避免重复构建相似的对象,提高了效率。
三、使用原型模式需要注意的问题
在使用原型模式时,有几个注意事项需要考虑:
- 深克隆与浅克隆:原型模式涉及克隆对象,而克隆可以分为深克隆和浅克隆。浅克隆只复制对象的值和指针,而不复制指针指向的对象。深克隆会递归地复制对象及其关联的所有对象。在确定克隆的方式时,需要考虑对象间的关系和内存管理。
- 构造函数和初始化:克隆对象不会调用构造函数,它是通过复制现有对象的内部状态来创建新对象。因此,如果在构造函数中进行了某些初始化操作,克隆对象可能会缺少这些初始化。需要确保对象的状态在克隆后正确。
- 单例模式和原型模式的结合:如果一个类同时实现了单例模式和原型模式,就需要特别小心。单例模式要求一个类只有一个实例,而原型模式要求创建新对象。在这种情况下,需要权衡是否要同时支持这两种模式。
- 对象图的复杂性:如果对象之间有复杂的关联关系,如循环引用等,克隆对象可能会导致对象图的复制变得复杂。需要仔细考虑对象之间的关系,以及克隆操作的顺序。
- 性能开销:深克隆可能涉及递归复制对象的所有关联对象,这可能导致性能开销较大。在性能敏感的情况下,需要权衡是否使用原型模式。
总的来说,原型模式在需要创建相似对象且效率要求较高的场景下非常有用。然而,在使用时需要注意克隆的方式、构造函数、对象关系等细节,以确保正确性和可维护性。

相关文章:
【设计模式--原型模式(Prototype Pattern)
一、什么是原型模式 原型模式(Prototype Pattern)是一种创建型设计模式,它的主要目的是通过复制现有对象来创建新的对象,而无需显式地使用构造函数或工厂方法。这种模式允许我们创建一个可定制的原型对象,然后通过复制…...
初识 Redis
初识 Redis 1 认识NoSQL1.1 结构化与非结构化1.2 关联和非关联1.3 查询方式1.4. 事务1.5 总结 2 Redis 概述2.1 应用场景2.2 特性 3 Resis 全局命令4 Redis 基本数据类型4.1 String4.1.1 常用命令4.1.2 命令的时间复杂度4.1.3 使用场景 4.2 Hash4.2.1 常用命令4.2.2 命令的时间…...
php灵异事件,啥都没干数据变了?
这篇文章也可以在我的博客查看 搞WordPress,难免跟php打交道 然而这弱类型语言实在坑有点多 这不今儿又踩了个大坑直接时间-1😅 问题 话不多说直接上代码 <?php $items [1,2];foreach ($items as &$item) {/*empty loop*/} print_r($items)…...
【ffmpeg】基于需要使用videocapture的opencv编译配置(C++)
目录 配置简介ffmpeg源码编译方法记录gstreamer命令行安装方法opencv的编译项记录 配置简介 opencv使用videocapture读取视频流时,需要借助底层的ffmpeg库。如果不能正确编译,会报错,现记录正确编译配置方法。 ffmpeg源码编译方法记录 ope…...
Redisson分布式锁 原理源码 分析
# 基于setnx实现的分布式锁存在的问题: # 为了解决上面的问题,可以用Redisson # Redisson入门 # Redisson可重入锁原理 获取锁的Lua脚本: 释放锁的Lua脚本: # 锁重试原理分析 tryLock()底层代码分析 tim…...
Cocos独立游戏开发框架中的事件管理器
引言 本系列是《8年主程手把手打造Cocos独立游戏开发框架》,欢迎大家关注分享收藏订阅。在独立游戏开发中,事件管理器是一个不可或缺的组件。它为开发者提供了一种灵活的方式来处理游戏内部各种状态变化和用户交互,实现模块之间的解耦和通信…...
keepalived+haproxy 搭建高可用高负载高性能rabbitmq集群
一、环境准备 1. 我这里准备了三台centos7 虚拟机 主机名主机地址软件node-01192.168.157.133rabbitmq、erlang、haproxy、keepalivednode-02192.168.157.134rabbitmq、erlang、haproxy、keepalivednode-03192.168.157.135rabbitmq、erlang 2. 关闭三台机器的防火墙 # 关闭…...
网络安全(黑客)零基础自学
网络安全是什么? 网络安全,顾名思义,网络上的信息安全。 随着信息技术的飞速发展和网络边界的逐渐模糊,关键信息基础设施、重要数据和个人隐私都面临新的威胁和风险。 网络安全工程师要做的,就是保护网络上的信息安…...
如何把本地项目上传github
一、在gitHub上创建新项目 【1】点击添加()-->New repository 【2】填写新项目的配置项 Repository name:项目名称 Description :项目的描述 Choose a license:license 【3】点击确定,项目已在githu…...
跳跃游戏【贪心算法】
跳跃游戏 给你一个非负整数数组 nums ,你最初位于数组的 第一个下标 。数组中的每个元素代表你在该位置可以跳跃的最大长度。 判断你是否能够到达最后一个下标,如果可以,返回 true ;否则,返回 false 。在这里插入图片…...
vue2+element-ui 实现下拉框滚动加载
一、自定义滚动指令。 VUE.directive( el-select-loadmore: { bind(el, binding) { const SELECTWRAP_DOM el.querySelector(.el-select-dropdown .el-select-dropdown__wrap) SELECTWRAP_DOM.addEventListener(scroll, function () { /*…...
探索AIGC人工智能(Midjourney篇)(二)
文章目录 利用Midjourney进行LOGO设计 用ChatGPT和Midjourney的AI绘画,制作儿童绘本故事 探索Midjourney换脸艺术 添加InsightFaceSwap机器人 Midjourney打造专属动漫头像 ChatGPT Midjourney画一幅水墨画 Midjourney包装设计之美 Midjourney24节气海报插画…...
01-Flask-简介及环境准备
Flask-简介及环境准备 前言简介特点Flask 与 Django 的比较环境准备 前言 本篇来介绍下Python的web框架–Flask。 简介 Flask 是一个轻量级的 Web 框架,使用 Python 语言编写,较其他同类型框架更为灵活、轻便且容易上手,小型团队在短时间内…...
【Git游戏】远程分支
origin/<branch> 远程分支在本地以 origin/<branch>格式存在,他指向上次和远程分支通过时的记录 git checkout origin/<branch> 会出现HEAD分离的情况 与远程通讯 git fetch —— 从远端获取数据(实际上将本地仓库中的远程分支更新…...
Day07-ElementUI
Day02-ElementUI 一 菜单设计 1 静态菜单 a 在components文件夹中新建一个组件Menu.vue <template><div class="menu-wrap"><el-menuclass="el-menu-vertical-demo"background-color="#031627"text-color="#fff"ac…...
【Go 基础篇】Go语言中的defer和recover:优雅处理错误
Go语言以其简洁、高效和强大的特性受到了开发者的热烈欢迎。在错误处理方面,Go语言提供了一种优雅的机制,即通过defer和recover组合来处理恐慌(panic)错误。本文将详细介绍Go语言中的defer和recover机制,探讨其工作原理…...
4.15 TCP Keepalive 和 HTTP Keep-Alive 是一个东西吗?
目录 HTTP 的 Keep-Alive TCP 的 Keepalive 总结: HTTP的Keep-Alive,是应用层(用户态)实现的,称为HTTP长连接; TCP的Keepalive,是由TCP层(内核态)实现的,…...
如何在VSCode中将html文件打开到浏览器
天行健,君子以自强不息;地势坤,君子以厚德载物。 每个人都有惰性,但不断学习是好好生活的根本,共勉! 文章均为学习整理笔记,分享记录为主,如有错误请指正,共同学习进步。…...
2022年03月 C/C++(四级)真题解析#中国电子学会#全国青少年软件编程等级考试
第1题:拦截导弹 某国为了防御敌国的导弹袭击, 发展出一种导弹拦截系统。 但是这种导弹拦截系统有一个缺陷: 虽然它的第一发炮弹能够到达任意的高度,但是以后每一发炮弹都不能高于前一发的高度。 某天, 雷达捕捉到敌国的…...
五公里场地训练笔记(完整版)
由于考研和口罩等原因,停跑了比较长的时间。中长距离就是这样,修为尽失,大概是要从头开始了,不过还是要乐观的面对,CHEER UP! 翻看咕咚软件,以前的PB是21:12,在2017年9月…...
MODBUS TCP转CANopen 技术赋能高效协同作业
在现代工业自动化领域,MODBUS TCP和CANopen两种通讯协议因其稳定性和高效性被广泛应用于各种设备和系统中。而随着科技的不断进步,这两种通讯协议也正在被逐步融合,形成了一种新型的通讯方式——开疆智能MODBUS TCP转CANopen网关KJ-TCPC-CANP…...
【单片机期末】单片机系统设计
主要内容:系统状态机,系统时基,系统需求分析,系统构建,系统状态流图 一、题目要求 二、绘制系统状态流图 题目:根据上述描述绘制系统状态流图,注明状态转移条件及方向。 三、利用定时器产生时…...
【配置 YOLOX 用于按目录分类的图片数据集】
现在的图标点选越来越多,如何一步解决,采用 YOLOX 目标检测模式则可以轻松解决 要在 YOLOX 中使用按目录分类的图片数据集(每个目录代表一个类别,目录下是该类别的所有图片),你需要进行以下配置步骤&#x…...
今日科技热点速览
🔥 今日科技热点速览 🎮 任天堂Switch 2 正式发售 任天堂新一代游戏主机 Switch 2 今日正式上线发售,主打更强图形性能与沉浸式体验,支持多模态交互,受到全球玩家热捧 。 🤖 人工智能持续突破 DeepSeek-R1&…...
JVM暂停(Stop-The-World,STW)的原因分类及对应排查方案
JVM暂停(Stop-The-World,STW)的完整原因分类及对应排查方案,结合JVM运行机制和常见故障场景整理而成: 一、GC相关暂停 1. 安全点(Safepoint)阻塞 现象:JVM暂停但无GC日志,日志显示No GCs detected。原因:JVM等待所有线程进入安全点(如…...
如何理解 IP 数据报中的 TTL?
目录 前言理解 前言 面试灵魂一问:说说对 IP 数据报中 TTL 的理解?我们都知道,IP 数据报由首部和数据两部分组成,首部又分为两部分:固定部分和可变部分,共占 20 字节,而即将讨论的 TTL 就位于首…...
SAP学习笔记 - 开发26 - 前端Fiori开发 OData V2 和 V4 的差异 (Deepseek整理)
上一章用到了V2 的概念,其实 Fiori当中还有 V4,咱们这一章来总结一下 V2 和 V4。 SAP学习笔记 - 开发25 - 前端Fiori开发 Remote OData Service(使用远端Odata服务),代理中间件(ui5-middleware-simpleproxy)-CSDN博客…...
LeetCode - 199. 二叉树的右视图
题目 199. 二叉树的右视图 - 力扣(LeetCode) 思路 右视图是指从树的右侧看,对于每一层,只能看到该层最右边的节点。实现思路是: 使用深度优先搜索(DFS)按照"根-右-左"的顺序遍历树记录每个节点的深度对于…...
【7色560页】职场可视化逻辑图高级数据分析PPT模版
7种色调职场工作汇报PPT,橙蓝、黑红、红蓝、蓝橙灰、浅蓝、浅绿、深蓝七种色调模版 【7色560页】职场可视化逻辑图高级数据分析PPT模版:职场可视化逻辑图分析PPT模版https://pan.quark.cn/s/78aeabbd92d1...
MacOS下Homebrew国内镜像加速指南(2025最新国内镜像加速)
macos brew国内镜像加速方法 brew install 加速formula.jws.json下载慢加速 🍺 最新版brew安装慢到怀疑人生?别怕,教你轻松起飞! 最近Homebrew更新至最新版,每次执行 brew 命令时都会自动从官方地址 https://formulae.…...
