1.4设计模式之原型模式(Prototype)

1.4.1 模式意图:

当实际开发中,需要在 运行期间 通过 已创建的实例,复制和自身一模一样(也可定制)的对象(类似于细胞分裂)。对于这种需求可以使用“原型模式”解决。

1.4.2 模式概念:

此模式属于创建型模式。用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象。

1.4.3 模式元素:

  • 原型类(Cell、LifeCycleLog等)
  • 复制接口(ICloneable)

1.4.4 代码示例:

A:基础示例

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using System;
using Custom.Log;
using System.Threading;

public class LifeCycleLog: ICloneable
{
    public string description;

    public object Clone()
    {
        return MemberwiseClone();
    }
}
public class Cell:ICloneable
{
    public int id;
    public string name;
    public LifeCycleLog lifeCycleLog;

    public Cell(int tempID,string tempName)
    {
        this.id = tempID;
        this.name = tempName;
        lifeCycleLog = new LifeCycleLog();
        Thread.Sleep(1000);
    }

    public object Clone()
    {
        Cell tempCell = new Cell(this.id, this.name);
        tempCell.lifeCycleLog.description = lifeCycleLog.description;
        return tempCell;
    }
    public void Log()
    {
        this.Log($"细胞ID:{id},细胞名称:{name},日志:{lifeCycleLog.description}");
    }
}

调用

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using Custom.Log;
using System.Threading;
using System.Threading.Tasks;
public class PrototypePatternComponent : MonoBehaviour
{
    void Start()
    {
        Task.Run(() => 
        {
            Cell Cell = new Cell(0, "编号00");
            Cell.lifeCycleLog.description = "生命周期日志00";
            Cell.Log();
            this.Log("【准备复制】");
            Thread.Sleep(500);

            for (int i = 0; i < 5; i++)
            {
               Cell tempCell =  (Cell)Cell.Clone();
                tempCell.Log();
            }
            this.Log("【复制完毕】");
        });
    }
}

打印日志

这样我们最粗糙的原型模式就算是完成了,是不感觉很简单呢。为什么说粗糙呢?因为还有更高效的拷贝方式。

B:优化示例
使用C#自带的MemberwiseClone函数。只需要继承ICloneable接口将原来的Clone函数稍加改动,速度即可快的飞起~

    public object Clone()
    {
        return MemberwiseClone();
    }

打印日志

MemberwiseClone函数对应的文档中我们可知道,这种复制属于浅拷贝,这时笔者需要引入两个概念,浅拷贝深拷贝

【浅拷贝】
如下图:通过浅拷贝创建一个新对象,然后将当前对象的非静态字段复制到新对象。如果字段是值类型,将执行字段的逐位复制; 对于引用类型,将复制引用,但不复制引用对象; 因此从原始对象复制的是值类型和引用类型的引用。在C#和VB.NET中,浅层复制由对象MemberwiseClone()方法完成。

【深拷贝】
如下图:深拷贝创建一个新对象,然后将当前对象的非静态字段复制到新对象。如果字段是值类型将执行字段的逐位复制。如果字段是引用类型将执行引用对象的新副本。

C:浅拷贝示例

    void Start()
    {
        Cell cell = new Cell(0, "编号00");
        cell.lifeCycleLog.description = "生命周期日志00";

        Cell cell01 = (Cell)cell.Clone();
        cell01.id = 1;
        cell01.name = "编号01";
        cell01.lifeCycleLog.description = "改动后的生命周期日志";

        cell.Log();
        cell01.Log();
    }

打印信息

由示例可看出,笔者只更改了cell01中的lifeCycleLog的描述,但所有的描述都发生了更改,这是因为使用了浅拷贝的缘故。

D:深拷贝示例

    public object DeepClone()
    {
        Cell tempCell = (Cell)MemberwiseClone();
        tempCell.lifeCycleLog = (LifeCycleLog)this.lifeCycleLog.Clone();
        return tempCell;
    }

打印日志

需要深拷贝的话我们可以这么写,既保证了速度又进行了深度的拷贝。

当然深拷贝还有以下几种方式

E:String类型特例

不知道这回大家是否明白为什么会造成lifeCycleLog描述一致的原因呢?最初笔者曾认为浅拷贝是Cell cell01 = cell;这种形式的,现在看并没有那么“浅”。
但是也有人可能会问,string类型也是引用类型,而且stirng有特殊的享元模式,为什么没有一同改变呢?
这就要看下String类的注解:字符串是用于表示文本的字符的有序集合。 String对象是表示字符串 System.Char对象的有序集合。System.Char 对象对应于 UTF-16 代码单元。 String对象的值是 System.Char对象的顺序集合的内容,并且该值是不可变的(即它是只读的)。 有关字符串不可变性的详细信息,请参阅 "不可变性和 StringBuilder 类" 一节。 内存中 String对象的最大大小为2GB 或大约1000000000个字符。

1.4.5 写法对比:

1.4.6 模式分析:

  • 创建该类型的新实例而不调用任何构造函数,并从源对象复制每个实例字段值,避免构造函数的约束从而提高性能。
  • 必须实现 Cloneable 接口。
  • 需要注意深拷贝和复杂的引用类型(循环引用)。

1.4.7 应用场景:

  • 多用于创建复杂的或耗时的实例,因为在这种情况下,复制一个已经存在的实例可以使程序运行的更高效。
  • 创建值相等,只是命名不一样的同类数据。

1.4.8 小结:

  • 原型模式其实就是一个对象再创建另一个可定制的对象,而且不需要知道任何创建的细节。
  • 从已经存在的实例来返回新的实例,而不是新建实例。

如果大家在项目中的需求是从一个已有的实体,复制出多个一模一样的实体,且初始化比较耗时,原型模式是个不错的选择~


更多设计模式详见:设计模式全家桶

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,794评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,050评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,587评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,861评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,901评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,898评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,832评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,617评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,077评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,349评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,483评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,199评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,824评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,442评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,632评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,474评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,393评论 2 352

推荐阅读更多精彩内容

  • 设计模式概述 在学习面向对象七大设计原则时需要注意以下几点:a) 高内聚、低耦合和单一职能的“冲突”实际上,这两者...
    彦帧阅读 3,741评论 0 14
  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,094评论 1 32
  • 1大同小异的工作周报 Sunny软件公司一直使用自行开发的一套OA (Office Automatic,办公自动化...
    justCode_阅读 1,153评论 0 3
  • 今天参加日更的第一天,说实话没有每天记日记的习惯。 以前小的时候还喜欢有什么事情都记录下来,后来,记的日记要给家长...
    谁说未来可期阅读 201评论 0 0
  • 那些活得幸福的人,不是忘记了,而是放下了;不是认怂了,而是看开了;不是糊涂了,而是懂得了。 余生太快,很多事情经不...
    光辉岁月_ffed阅读 127评论 0 0