概述

描述

  • 使用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象。如孙悟空猴毛分身、鸣人影之分身、剑光分化、无限剑制

  • 原型模式是一种创建型设计模式,允许一个对象再创建另外一个可定制的对象,无需知道如何创建的细节。

  • 原型模式工作原理是:通过将一个原型对象传给那个要创建的对象,这个要创建的对象通过请求原型对象拷贝它们自己来实施创建对象,即对象Clone()

  • 浅拷贝概念

    • 对象语句类型是基本数据类型的成员变量,浅拷贝会直接进行值传递,也就是将该属性值直接复制一份给新对象。
    • 对于数据类型是引用类型的成员变量,那么浅拷贝会进行引用传递,也就是只会讲改成员变量的引用值(内存地址)复制一份给新的对象,实际上两个对象的该成员变量都指向同一个实例,在这种情况下修改一个对象的变量会影响到另一个对象的该成员变量的值。
  • 深拷贝概念

    • 复制对象所有基本数据类型的成员变量值
    • 为所有引用数据类型的成员白能量申请存储空间,并赋值每个引用数据类型成员变量所引用的对象,也就是说对象进行深拷贝要对整个对象进行拷贝。
    • 深拷贝的两种实现方式,1.重写clone方法实现深拷贝,2.通过对象序列化和反序列化实现深拷贝(推荐)

套路

  • Prototype(抽象原型类)

    它是声明克隆方法的接口,是所有具体原型类的公共父类,可以是抽象类也可以是接口,甚至还可以是具体实现类。
  • ConcretePrototype(具体原型类)

    它实现在抽象原型类中声明的克隆方法,在克隆方法中返回自己的一个克隆对象
  • Client(客户类):让一个原型对象克隆自身从而创建一个新的对象,在客户类中只需要直接实例化或通过工厂方法等方式创建一个原型对象,再通过调用该对象的克隆方法即可得到多个相同的对象

使用场景

  • 有性能要求的场景,如资源优化
  • 类初始化需要消耗过多的资源、如数据、硬盘、cpu、网络资源等等
  • 实例化新对象需要繁琐的准备步骤
  • 需要方便复制的时候,如拖动预设到场景中、场景中物体快速复制

优缺点

  • 优点

    • 由于客户类针对抽象原型类Prototype编程,因此用户可以根据需要选择具体原型类,系统具有较好的可扩展性,增加或更换具体原型类都很方便。
    • 当创建新的对象实例较为复杂时,使用原型模式可以简化对象的创建过程,通过复制一个已有的实例可以提高新实例的创建效率。
    • 可以使用深复制的方式保存对象的状态。将对象复制一份并将其状态保存起来,以便于在使用的时候使用,比如恢复到某一个历史状态,可以辅助实现撤销操作。
  • 缺点
    • 需要为每一个类配备一个克隆方法,而且该克隆方法位于一个类的内部,当对已有的类进行改造时,需要修改源代码,违背了开闭原则。
    • 为了支持深复制,当对象之间存在多重嵌套引用关系时,每一层对象都必须支持深复制,实现起来可能比较麻烦。

UE4 实践

  • 写一个角色生成器、生成克隆角色

  • 原型抽象类、具体类 —— 怪物、Ghost、Devil

    // 抽象原型类 —— 怪物
    UCLASS(Abstract)
    class DESIGNPATTERNS_API UMonsterPrototype : public UObject
    {
    GENERATED_BODY()
    public:
    // 克隆函数
    virtual UMonsterPrototype* Clone() {
    UE_LOG(LogTemp, Error, TEXT("Please implement this!"));
    return nullptr;
    } // 展示信息
    virtual void ShowInfo() {
    UE_LOG(LogTemp, Warning, TEXT(__FUNCTION__" %s [Health]%d, [Speed]%d"), *this->GetName() ,m_pHealth, m_pSpeed);
    } protected:
    int32 m_pHealth = 100;
    int32 m_pSpeed = 30;
    }; // 具体产原型类 —— Ghost
    UCLASS(Blueprintable, BlueprintType)
    class DESIGNPATTERNS_API UGhost : public UMonsterPrototype
    {
    GENERATED_BODY()
    public: // 重载克隆函数
    virtual UMonsterPrototype* Clone() override {
    UGhost* CloneIns = NewObject<UGhost>();
    CloneIns->m_pHealth = m_pHealth;
    CloneIns->m_pSpeed = m_pSpeed;
    return CloneIns;
    }
    }; // 具体产原型类 —— Devil
    UCLASS(Blueprintable, BlueprintType)
    class DESIGNPATTERNS_API UDevil : public UMonsterPrototype
    {
    GENERATED_BODY()
    public:
    // 初始化数值
    UDevil() {
    m_pHealth = 120;
    m_pSpeed = 20;
    } // 重载克隆函数
    virtual UMonsterPrototype* Clone() override {
    UDevil* CloneIns = NewObject<UDevil>();
    CloneIns->m_pHealth = m_pHealth;
    CloneIns->m_pSpeed = m_pSpeed;
    CloneIns->m_pAttack = m_pAttack;
    return CloneIns;
    } // 展示信息
    virtual void ShowInfo() override {
    UE_LOG(LogTemp, Warning, TEXT(__FUNCTION__" %s [Health]%d, [Speed]%d, [Attack] %d"), *this->GetName(), m_pHealth, m_pSpeed, m_pAttack);
    }
    protected:
    int32 m_pAttack = 100;
    };
  • 创建工厂类 —— 怪物生成器

    // 工厂类 —— 怪物生成器
    UCLASS(Blueprintable, BlueprintType)
    class DESIGNPATTERNS_API UMonsterSpawner : public UObject
    {
    GENERATED_BODY()
    public: // 生成新怪物,使用模板,避免针对每种怪物都要写一遍
    template <class T>
    T* SpawnMonster() {
    return NewObject<T>();
    } // 克隆怪物
    UMonsterPrototype* SpawnMonster(UMonsterPrototype* pMonsterClass) {
    return pMonsterClass->Clone();
    }
    };
  • 调用测试

    // 调用测试用的Actor
    UCLASS()
    class DESIGNPATTERNS_API AMonsterSpawnerActor : public AActor
    {
    GENERATED_BODY()
    public: void BeginPlay() override { // 创建工厂
    UMonsterSpawner* MonsterSpawner = NewObject<UMonsterSpawner>(); // 第一次创建 Ghost
    UGhost* Ghost = MonsterSpawner->SpawnMonster<UGhost>();
    // 克隆 Ghost
    UGhost* Ghost_Copy1 = Cast<UGhost>(MonsterSpawner->SpawnMonster(Ghost));
    Ghost->ShowInfo();
    Ghost_Copy1->ShowInfo(); // 第一次创建 Devil
    UDevil* Devil =MonsterSpawner->SpawnMonster<UDevil>();
    // 克隆 Devil
    UDevil* Devil_Copy1 = Cast<UDevil>(MonsterSpawner->SpawnMonster(Devil));
    Devil->ShowInfo();
    Devil_Copy1->ShowInfo();
    }
    };
  • 调式输出

    LogTemp: Warning: UMonsterPrototype::ShowInfo Ghost_0 [Health]100, [Speed]30
    LogTemp: Warning: UMonsterPrototype::ShowInfo Ghost_1 [Health]100, [Speed]30
    LogTemp: Warning: UDevil::ShowInfo Devil_0 [Health]120, [Speed]20, [Attack] 100
    LogTemp: Warning: UDevil::ShowInfo Devil_1 [Health]120, [Speed]20, [Attack] 100

为数据模型构建原型

  • 在数据建模时,同样可以运用原型模式

  • 给对象声明一个prototype属性,该属性指定另外一个对象,如果访问的属性不在此对象内部,就去prototype指定的对象查找

    {
    "name": "goblin grunt",
    "minHealth": 20,
    "maxHealth": 30,
    "resists": ["cold", "poison"],
    "weaknesses": ["fire", "light"]
    }
    {
    "name": "goblin wizard",
    "prototype": "goblin grunt",
    "spells": ["fire ball", "lighting bolt"]
    }
    {
    "name": "goblin archer",
    "prototype": "goblin grunt",
    "attacks": ["short bow"]
    }

参考

【UE4 设计模式】原型模式 Prototype Pattern的更多相关文章

  1. C#设计模式——原型模式(Prototype Pattern)

    一.概述 在软件开发中,经常会碰上某些对象,其创建的过程比较复杂,而且随着需求的变化,其创建过程也会发生剧烈的变化,但他们的接口却能比较稳定.对这类对象的创建,我们应该遵循依赖倒置原则,即抽象不应该依 ...

  2. 设计模式——原型模式(Prototype Pattern)

    原型模式:用原型实例制定创建对象的种类,并且通过拷贝这些原型创建新的对象. UML 图: 原型类: package com.cnblog.clarck; /** * 原型类 * * @author c ...

  3. Net设计模式实例之原型模式( Prototype Pattern)

    一.原型模式简介(Brief Introduction) 原型模式(Prototype Pattern):用原型实例指定创建对象的种类,并通过拷贝这些原型创建新的对象. Specify the kin ...

  4. 乐在其中设计模式(C#) - 原型模式(Prototype Pattern)

    原文:乐在其中设计模式(C#) - 原型模式(Prototype Pattern) [索引页][源码下载] 乐在其中设计模式(C#) - 原型模式(Prototype Pattern) 作者:weba ...

  5. 设计模式系列之原型模式(Prototype Pattern)——对象的克隆

    说明:设计模式系列文章是读刘伟所著<设计模式的艺术之道(软件开发人员内功修炼之道)>一书的阅读笔记.个人感觉这本书讲的不错,有兴趣推荐读一读.详细内容也可以看看此书作者的博客https:/ ...

  6. 二十四种设计模式:原型模式(Prototype Pattern)

    原型模式(Prototype Pattern) 介绍用原型实例指定创建对象的种类,并且通过拷贝这个原型来创建新的对象.示例有一个Message实体类,现在要克隆它. MessageModel usin ...

  7. PHP设计模式 原型模式(Prototype)

    定义 和工厂模式类似,用来创建对象.但实现机制不同,原型模式是先创建一个对象,采用clone的方式进行新对象的创建. 场景 大对象的创建. 优点 1.可以在运行时刻增加和删除产品 2.可以改变值或结构 ...

  8. 2.6 《硬啃设计模式》第8章 复制不是很难 - 原型模式(Prototype Pattern)

    案例: 某即时战略游戏,你训练出来各种很强的战士. 为了增加游戏的可玩性,增加了一种复制魔法.实施该魔法,可以复制任意的战士. 你会怎样考虑这个设计? 在继续阅读之前,请先认真思考并写出你的设计,这样 ...

  9. python 设计模式之原型模式 Prototype Pattern

    #引入 例子1: 孙悟空拔下一嘬猴毛,轻轻一吹就会变出好多的孙悟空来. 例子2:寄个快递下面是一个邮寄快递的场景:“给我寄个快递.”顾客说.“寄往什么地方?寄给……?”你问.“和上次差不多一样,只是邮 ...

随机推荐

  1. 【开发工具】Postman保姆级入门教程

    目录 一.简单使用 1. 创建命名空间 2. 创建新集合 3. 按模块整理接口 二.使用环境变量 1. 创建环境与环境变量 2. 使用环境变量 3. 登录后自动更新环境变量 转载请注明出处 一.简单使 ...

  2. 20210826 Lighthouse,Miner,Lyk Love painting,Revive

    考场 T1 这不裸的容斥 T2 这不裸的欧拉路,先从奇数度点开始走,走不了就传送 T3 什么玩意,暴力都不会 T4 点分树??? 仔细想了一波,发现 T1 T2 都好做,T3 二分答案后可以暴力贪心, ...

  3. Linux触摸驱动分析

    测试平台 宿主机平台:Ubuntu 12.04.4 LTS 目标机:Easy-ARM IMX283 目标机内核:Linux 2.6.35.3 触摸屏基础知识 一.结构 上图是电阻触摸屏的一个侧面剖视图 ...

  4. autoCAD绘制简单三维立体图形

    第一步: 首先绘制一个简单的(封闭的)二维图形: 第二步: 变换观察视角,比如修改为 变换后的视角: 第三步: 选中闭合图形边框,使用组合键"ctrl + shift + E",然 ...

  5. [第二篇]——Docker 架构之Spring Cloud直播商城 b2b2c电子商务技术总结

    Docker 架构 Docker 包括三个基本概念: 镜像(Image):Docker 镜像(Image),就相当于是一个 root 文件系统.比如官方镜像 ubuntu:16.04 就包含了完整的一 ...

  6. Java学习笔记--面对对象OOP

    面向对象编程 OOP 面向对象&面向过程 面向对象编程的本质:以类的方式组织代码,以对象的方法组织数据 面向对象编程的三大特征: 封装 继承 多态 方法 静态方法 通过 static 关键词说 ...

  7. go新建一个工程

    使用go mod 可以在任何地方新建工程 工程目录 main.go   //引用子包必须格式"工程目录/子包" go.mod 子包 编译工程: go build

  8. Linux中不用用户可以使用相同的uid

    usermod -u 513 -o  tom 使得用户tom可以使用uid等于513,即使513已经被其他用户使用了

  9. Git - 命令行 常用

    一.合并其他分支的commit(A分支中的commit合并至B分支) 切换到A分支,查询commit历史命令行 : $ git log 复制要合并的commit id (如:663802dfb121e ...

  10. Docker系列(14)- Portainer可视化面板安装

    官网 https://documentation.portainer.io/v2.0-be/deploy/beinstalldocker/ 可视化 portainer docker run -d -p ...