23种设计模式-原型(Prototype)设计模式

news2024/12/27 17:04:48

文章目录

  • 一.什么是原型设计模式?
  • 二.原型模式的特点
  • 三.原型模式的结构
  • 四.原型模式的优缺点
  • 五.原型模式的 C++ 实现
  • 六.原型模式的 Java 实现
  • 七. 代码解析
  • 八.总结

类图: 原型设计模式类图

一.什么是原型设计模式?

原型模式(Prototype Pattern) 是一种创建型设计模式,它通过复制现有对象来生成新对象,而不是通过实例化类来创建。这种模式基于对象的克隆机制,适用于需要频繁创建对象的场景,可以提高性能并减少复杂的初始化过程。

二.原型模式的特点

  • 克隆对象:通过复制已有对象生成新对象,而非通过构造函数创建。
  • 性能优化:避免复杂对象创建的高开销。
  • 灵活性:可以动态添加或删除原型,而不影响其他对象。

三.原型模式的结构

  • Prototype(抽象原型类):定义了克隆方法,通常是 clone() 方法。
  • ConcretePrototype(具体原型类):实现抽象原型类的克隆方法。
  • Client(客户端):调用克隆方法创建新对象,而无需关心对象的具体类型。
    原型设计模式类图

四.原型模式的优缺点

  • 优点:
    • 性能优势:克隆比重新创建对象效率更高。
    • 动态性:无需编译时依赖类,可以在运行时动态生成对象。
    • 避免依赖性:通过克隆机制,可以避免直接依赖类的构造函数。
  • 缺点:
    • 复杂性增加:需要正确实现克隆方法,尤其是深拷贝场景。
    • 资源消耗:可能会为每个可克隆的对象维护克隆逻辑,导致类职责增多。

五.原型模式的 C++ 实现

#include <iostream>
#include <string>
#include <memory> // for smart pointers
using namespace std;

// 抽象原型类
class Prototype {
public:
    virtual ~Prototype() = default;

    // 克隆方法(纯虚函数)
    virtual unique_ptr<Prototype> clone() const = 0;

    virtual void display() const = 0;
};

// 具体原型类1
class ConcretePrototypeA : public Prototype {
private:
    string data; // 模拟一些状态
public:
    ConcretePrototypeA(string data) : data(data) {}

    // 克隆方法
    unique_ptr<Prototype> clone() const override {
        return make_unique<ConcretePrototypeA>(*this); // 调用拷贝构造函数
    }

    void display() const override {
        cout << "ConcretePrototypeA with data: " << data << endl;
    }
};

// 具体原型类2
class ConcretePrototypeB : public Prototype {
private:
    int value; // 模拟一些状态
public:
    ConcretePrototypeB(int value) : value(value) {}

    // 克隆方法
    unique_ptr<Prototype> clone() const override {
        return make_unique<ConcretePrototypeB>(*this); // 调用拷贝构造函数
    }

    void display() const override {
        cout << "ConcretePrototypeB with value: " << value << endl;
    }
};

// 客户端代码
void ClientCode(const Prototype& prototype) {
    auto clonedObject = prototype.clone(); // 克隆一个对象
    clonedObject->display();               // 显示克隆对象的内容
}

int main() {
    // 创建原型对象
    ConcretePrototypeA prototypeA("PrototypeA Data");
    ConcretePrototypeB prototypeB(42);

    cout << "Cloning ConcretePrototypeA:" << endl;
    ClientCode(prototypeA);

    cout << "\nCloning ConcretePrototypeB:" << endl;
    ClientCode(prototypeB);

    return 0;
}

六.原型模式的 Java 实现

// 抽象原型类
abstract class Prototype implements Cloneable {
    public abstract Prototype clone();

    public abstract void display();
}

// 具体原型类A
class ConcretePrototypeA extends Prototype {
    private String data;

    public ConcretePrototypeA(String data) {
        this.data = data;
    }

    @Override
    public Prototype clone() {
        try {
            return (Prototype) super.clone();
        } catch (CloneNotSupportedException e) {
            throw new RuntimeException("Clone not supported!");
        }
    }

    @Override
    public void display() {
        System.out.println("ConcretePrototypeA with data: " + data);
    }
}

// 具体原型类B
class ConcretePrototypeB extends Prototype {
    private int value;

    public ConcretePrototypeB(int value) {
        this.value = value;
    }

    @Override
    public Prototype clone() {
        try {
            return (Prototype) super.clone();
        } catch (CloneNotSupportedException e) {
            throw new RuntimeException("Clone not supported!");
        }
    }

    @Override
    public void display() {
        System.out.println("ConcretePrototypeB with value: " + value);
    }
}

// 客户端代码
public class PrototypePatternExample {
    public static void main(String[] args) {
        // 创建原型对象
        ConcretePrototypeA prototypeA = new ConcretePrototypeA("PrototypeA Data");
        ConcretePrototypeB prototypeB = new ConcretePrototypeB(42);

        System.out.println("Cloning ConcretePrototypeA:");
        Prototype clonedA = prototypeA.clone();
        clonedA.display();

        System.out.println("\nCloning ConcretePrototypeB:");
        Prototype clonedB = prototypeB.clone();
        clonedB.display();
    }
}

七. 代码解析

  • 抽象原型类(Prototype)
    • 定义了一个 clone() 方法,用于生成当前对象的副本。
    • 使用虚函数,使得派生类可以实现自己的克隆逻辑。
  • 具体原型类(ConcretePrototypeA 和 ConcretePrototypeB)
    • 实现了 clone() 方法,通过调用自身的拷贝构造函数完成对象的克隆。
    • ConcretePrototypeA 和 ConcretePrototypeB 模拟了不同的数据成员,演示了克隆不同类型对象的效果。
  • 客户端代码
    • 接收一个 Prototype 对象的引用,并通过调用 clone() 方法克隆出一个新对象。
    • 客户端无需关心具体的类,只需通过 Prototype 接口调用克隆方法即可。

八.总结

 原型模式通过克隆的方式创建新对象,而不依赖于类的构造函数。这种模式在需要频繁创建对象、或者需要动态生成对象的场景中非常有用。通过正确实现 clone() 方法,可以有效提高程序的灵活性和性能。然而,在实现复杂对象的深拷贝时,需要特别注意对象之间的依赖关系和资源管理,以避免潜在的错误。
应用场景:

  • 复杂对象的创建:初始化需要耗费大量资源的对象可以通过克隆来生成新对象。
  • 性能优化:避免使用构造函数或工厂方法重复创建对象。
  • 动态对象管理:需要动态生成对象而不依赖具体类。

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

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

相关文章

基于大数据python 房屋价格数据分析预测可视化系统(源码+LW+部署讲解+数据库+ppt)

&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01; 很对人不知道选题怎么选 不清楚自己适合做哪块内容 都可以免费来问我 避免后期給自己答辩找麻烦 增加难度&#xff08;部分学校只有一次答辩机会 没弄好就延迟…

使用Grafana K6来测测你的系统负载能力

背景 近期我们有个号称会有很高很高并发的系统要上线&#xff0c;为了测试一下自己开发的系统的负载能力&#xff0c;准备了点海克斯科技&#xff0c;来看看抗不抗的住。 之前笔者写过用Apache JMeter进行压力测试的文章&#xff08;传送门&#x1f449;&#xff1a;https://…

Java 上机实践10(常用实用类)

&#xff08;大家好&#xff0c;今天分享的是Java的相关知识&#xff0c;大家可以在评论区进行互动答疑哦~加油&#xff01;&#x1f495;&#xff09; 目录 Plug&#xff1a;程序实现方法一&#xff08;记事本&#xff09; 方法二&#xff08;IDEA&#xff09; 实验一&…

28.UE5实现对话系统

目录 1.对话结构的设计&#xff08;重点&#xff09; 2.NPC对话接口的实现 2.1创建类型为pawn的蓝图 2.2创建对话接口 3.对话组件的创建 4.对话的UI设计 4.1UI_对话内容 4.2UI_对话选项 4.3UI_对话选项框 5.对话组件的逻辑实现 通过组件蓝图&#xff0c;也就是下图中的…

混沌工程/混沌测试/云原生测试/云平台测试

背景 私有云/公有云/混合云等具有复杂&#xff0c;分布式&#xff0c;环境多样性等特点&#xff0c;许多特殊场景引发的线上问题很难被有效发现。所以需要引入混沌工程&#xff0c;建立对系统抵御生产环境中失控条件的能力以及信心&#xff0c;提高系统面对未知风险得能力。 …

OpenMP出现Stack Overflow及其疑问

今天对着《OpenMP核心技术指南》练习OpenMP&#xff0c;其中一个案例: #include <stdio.h> #include <math.h> #include <omp.h>#define ITER 100000000void main() {int i;double A[ITER];for (i 0; i < ITER; i)A[i] 2.0 * i;#pragma omp parallel{/…

小F的矩阵值调整

问题描述 小F得到了一个矩阵。如果矩阵中某一个格子的值是偶数&#xff0c;则该值变为它的三倍&#xff1b;如果是奇数&#xff0c;则保持不变。小F想知道调整后的矩阵是什么样子的。 测试样例 样例1&#xff1a; 输入&#xff1a;a [[1, 2, 3], [4, 5, 6]] 输出&#xff1a…

【Python网络爬虫笔记】5-(Request 带参数的get请求) 爬取豆瓣电影排行信息

目录 1.抓包工具查看网站信息2.代码实现3.运行结果 1.抓包工具查看网站信息 请求路径 url:https://movie.douban.com/typerank请求参数 页面往下拉&#xff0c;出现新的请求结果&#xff0c;参数start更新&#xff0c;每次刷新出20条新的电影数据 2.代码实现 # 使用网络爬…

JiaJia-CP-1,2,3的WP(2)

一.JiaJia-CP-2 一看题目&#xff0c;聊天软件&#xff0c;用的什么聊天软件直接userassist看运行过什么程序 vol -f JiaJia_Co.raw --profileWin7SP1x64 userassist 发现Telegram.exe(小飞机) 可能性很大啊(真是个摸鱼大神) 除此之外&#xff0c;filescan也能看到&#xff0…

群控系统服务端开发模式-应用开发-前端邮箱短信通道开发

一、添加视图 在根目录下src文件夹下views文件夹下param文件夹下emailsms文件夹下&#xff0c;新建index.vue&#xff0c;代码如下 <template><el-tabs type"border-card"><el-tab-pane v-if"$store.getters.butts.includes(ParamEmailsmsIndex…

C/C++ 数据结构与算法【线性表】 顺序表+链表详细解析【日常学习,考研必备】带图+详细代码

1&#xff09;线性表的定义 线性表&#xff08;List&#xff09;&#xff1a;零个或多个数据元素的有限序列。 线性表的数据集合为{a1,a2,…,an}&#xff0c;假设每个元素的类型均为DataType。其中&#xff0c;除第一个元素a1外&#xff0c;每一个元素有且只有一个直接前驱元素…

高效特征选择:优化机器学习的嵌入式方法指南

高效特征选择&#xff1a;优化机器学习的嵌入式方法指南 文章目录 一、说明二、特征选择方法三、嵌入式方法四、Lasso正则化五、Python 中的 Lasso 实现六、决策树的特征重要性七、Python 实现八、嵌入式方法和递归特征消除九、结论 一、说明 假设您正在处理一个大型数据集&am…

系统架构:MVVM

引言 MVVM 全称 Model-View-ViewModel&#xff0c;是在 MVP&#xff08;Model-View-Presenter&#xff09;架构模式基础上的进一步演进与优化。MVVM 与 MVP 的基本架构相似&#xff0c;但 MVVM 独特地引入了数据双向绑定机制。这一创新机制有效解决了 MVP 模式中 Model 与 Vie…

家校通小程序实战教程04教师管理

目录 1 创建数据源2 搭建管理后台3 搭建查询条件4 功能测试总结 我们上一篇介绍了如何将学生加入班级&#xff0c;学生加入之后就需要教师加入了。教师分为任课老师和班主任&#xff0c;班主任相当于一个班级的管理员&#xff0c;日常可以发布各种任务&#xff0c;发布接龙&…

cesium 3Dtiles变量

原本有一个变亮的属性luminanceAtZenith&#xff0c;但是新版本的cesium没有这个属性了。于是 let lightColor 3.0result._customShader new this.ffCesium.Cesium.CustomShader({fragmentShaderText:void fragmentMain(FragmentInput fsInput, inout czm_modelMaterial mate…

SpringBoot小知识(3):热部署知识

一、热部署 热部署是一个非常消耗内存的机制&#xff0c;在实际大型项目开发中几乎用不到&#xff0c;只有小型项目或者分模块或者不停机更新的时候才会用到&#xff0c;仁者见仁智者见智。 1.1 什么是热部署&#xff1f; 热部署是指在不停止应用程序或服务器的情况下&#xf…

vscode切换anaconda虚拟环境解释器不成功

问题&#xff1a; 切换解释器之后运行代码还是使用的原来的解释器 可以看到&#xff0c;我已经切换了“nlp”解释器&#xff0c;我的nltk包只在“nlp”环境下安装了&#xff0c;但是运行代码依然是"torch"解释器&#xff0c;所以找不到“nltk”包。 在网上找了各种…

widows下永久修改python的pip 配置文件

通过cmd永久修改pip 镜像源&#xff1a; 在cmd中输入&#xff1a; pip config set global.index-url https://pypi.tuna.tsinghua.edu.cn/simple会在"C:\Users\Administrator\AppData\Roaming"目录下创建一个pip\pip.ini文件&#xff1a; 使用记事本打开pip.ini文件…

openssl使用哈希算法生成随机密钥

文章目录 一、openssl中随机数函数**OpenSSL 随机数函数概览**1. **核心随机数函数** **常用函数详解**1. RAND_bytes2. RAND_priv_bytes3. RAND_seed 和 RAND_add4. RAND_status **随机数生成器的熵池****常见用例****注意事项** 二、使用哈希算法生成随机的密钥 一、openssl中…

【Python网络爬虫笔记】6- 网络爬虫中的Requests库

一、概述 Requests 是一个用 Python 语言编写的、简洁且功能强大的 HTTP 库。它允许开发者方便地发送各种 HTTP 请求&#xff0c;如 GET、POST、PUT、DELETE 等&#xff0c;并且可以轻松地处理请求的响应。这个库在 Python 生态系统中被广泛使用&#xff0c;无论是简单的网页数…