原型模式(Prototype)

原型模式的结构 原型模式包含以下3个角色: •Prototype(抽象原型类) •ConcretePrototype(具体原型类) •Client(客户类)

浅克隆与深克隆 浅克隆(Shallow Clone):当原型对象被复制时,只复制它本身和其中包含的值类型的成员变量,而引用类型的成员变量并没有复制 深克隆(Deep Clone):除了对象本身被复制外,对象所包含的所有成员变量也将被复制

原型核心代码

    /// <summary>
    /// 班级
    /// </summary>
    [Serializable]
    public class Class
    {
        public int Num { get; set; }
        public string Remark { get; set; }
    }

    [Serializable]
    public class StudentPrototype
    {
        public int Id { get; set; }
        public string Name { get; set; }
        public Class Class { get; set; }

        private StudentPrototype() { }

        private static readonly StudentPrototype _studentPrototype = null;

        static StudentPrototype()
        {
            _studentPrototype = new StudentPrototype()
            {
                Id = 0,
                Name = "复制",
                Class = new Class()
                {
                    Num = 100,
                    Remark = "100"
                }
            };
        }
        public static StudentPrototype CreateInstanceClone()
        {
            StudentPrototype studentPrototype = (StudentPrototype)_studentPrototype.MemberwiseClone();
            //这样就是深clone,如果没有这段,Class实例就是浅克隆
            studentPrototype.Class = new Class()
            {
                Num = 1,
                Remark = "软谋高级班"
            };

            return studentPrototype;
        }

        public static StudentPrototype CreateInstanceSerialize()
        {
            //序列化方式创建克隆,需要保证被克隆的对象具有 serialize 特性标签
            return SerializeHelper.DeepClone<StudentPrototype>(_studentPrototype);
        }
    }  

原型模式的理解 原型模式在我的理解中时基于单例模式的一个扩展,在保证实例对象唯一的情况下,能快递new出不同的新实例 对外提供一个接口 CreateCloneInstance创建克隆对象

浅层次克隆

                Console.WriteLine("*****浅层次克隆Start**********");
                var box1 = StudentPrototype.CreateInstanceClone();
                Console.WriteLine($"Id={box1.Id} name={box1.Name}");
                var box2 = StudentPrototype.CreateInstanceClone();
                box2.Id = 10;
                box2.Name = "测试";
                Console.WriteLine($"Id={box2.Id} name={box2.Name}");
                Console.WriteLine("*****浅层次克隆End**********");

深层次克隆1 失败+成功方法

            {
                //对值类型的克隆修改值,不会影响克隆出的新对象的内容。但是引用类型的克隆有需要注意的点,Class是引用类型的值,具有Num和Remark俩个属性。如果使用下面的写法 box1.Class.Num=***,实际上源对象Class地址和克隆对象地址相同,修改一个相当于全部修改了,并没有实现我们想要的克隆效果
                Console.WriteLine("*****深层次克隆 Start**********");
                var box1 = StudentPrototype.CreateInstanceClone();
                box1.Class.Num = 10;
                box1.Class.Remark = "备注";
                Console.WriteLine($"Id={box1.Id} name={box1.Name} ClassNum={box1.Class.Num } Remark={box1.Class.Remark }");
                var box2 = StudentPrototype.CreateInstanceClone();
                box2.Id = 10;
                box2.Name = "测试";
                Console.WriteLine($"Id={box2.Id} name={box2.Name} ClassNum={box2.Class.Num } Remark={box2.Class.Remark }");
                Console.WriteLine("*****深层次克隆 End**********");
            }

深层次克隆2 序列化

   {
                //使用序列化的方式进行克隆原理,因为深层次克隆的主要问题是引用类型克隆时还是相同地址
                //我们先将初始对象序列化,在将得到的值进行反序列化,这样便能保证生成的对象存在不同的地址引用
                Console.WriteLine("*****深层次克隆2 序列化Start**********");
                var box1 = StudentPrototype.CreateInstanceSerialize();
                box1.Class.Num = 10;
                box1.Class.Remark = "备注";
                Console.WriteLine($"Id={box1.Id} name={box1.Name} ClassNum={box1.Class.Num } Remark={box1.Class.Remark }");
                var box2 = StudentPrototype.CreateInstanceSerialize();
                box2.Id = 10;
                box2.Name = "测试";
                Console.WriteLine($"Id={box2.Id} name={box2.Name} ClassNum={box2.Class.Num } Remark={box2.Class.Remark }");
                Console.WriteLine("*****深层次克隆2  序列化End**********");
            }

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏用户3030674的专栏

Java中Json解析

首先准备一个JSON格式的字符串 * String JsonStr = "{object:{persons:" + "[{name:'呵呵',im...

44420
来自专栏恰童鞋骚年

剑指Offer面试题:11.打印1到最大的n位数

  初看之下好像没有问题,但是其并没有考虑大数问题,有可能即使用整型(int)或长整型(long)都会溢出。

11820
来自专栏Android干货

Java数据解析---JSON

70370
来自专栏犀利豆的技术空间

Redis 的基础数据结构(三)对象

前两篇文章介绍了 Redis 的基本数据结构动态字符串,链表,字典,跳跃表,压缩链表,整数集合,但是使用过 Redis 的同学会发现,平时根本没有使用过这些数...

12820
来自专栏跟着阿笨一起玩NET

DataGridView绑定BindingList<T>带数据排序的类

本文章转载:http://yuyingying1986.blog.hexun.com/30905610_d.html

10610
来自专栏菩提树下的杨过

C#执行XSL转换

xsl 可方便的将一种格式的xml,转换成另一种格式的xml,参考下面的代码: using System; using System.IO; using Sys...

21090
来自专栏大内老A

通过实例模拟ASP.NET MVC的Model绑定的机制:集合+字典

在本系列的前面两篇文章(《简单类型+复杂类型》、《数组》)我们通过创建的实例程序模拟了ASP.NET MVC默认使用的DefaultModelBinder对简单...

30870
来自专栏GreenLeaves

反射基础入门篇

1、简介 Reflection,中文翻译为反射。.Net的应用程序由以下几个部分组成: a、程序集(assembly) b、模块(moudle) c、类型(cl...

19860
来自专栏林德熙的博客

win10 uwp unix timestamp 时间戳 转 DateTime

有时候需要把网络的 unix timestamp 转为 C# 的 DateTime ,在 UWP 可以如何转换?

7510
来自专栏码云1024

C#反射

32430

扫码关注云+社区

领取腾讯云代金券