zoukankan      html  css  js  c++  java
  • C# 之类复制 MemberwiseClone与Clone(深 浅 Clone)

       今天在《大话设计模式》上看了原型模式,里面对于深浅clone做了一定的介绍,本人觉得下面这篇文章对于处理深clone是提出了一种较好的方法。
     
       MemberwiseClone方法创建一个浅表副本,具体来说就是创建一个新对象,然后将当前对象的非静态字段复制到该新对象。如果字段是值类型的,则对该字段执行逐位复制。如果字段是引用类型,则复制引用但不复制引用的对象。因此,原始对象及其复本引用的是同一对象。

       为了实现深度复制,我们就必须遍历有相互引用的对象构成的图,并需要处理其中的循环引用结构。这无疑是十分复杂的。幸好借助.Net的序列化和反序列化机制,可以十分简单的深度Clone一个对象。原理很简单,首先将对象序列化到内存流中,此时对象和对象引用的所用对象的状态都被保存到内存中。.Net的序列化机制会自动处理循环引用的情况。然后将内存流中的状态信息反序列化到一个新的对象中。这样一个对象的深度复制就完成了。在原型设计模式中clone技术非常关键。

    下面的代码就是演示这个问题:

    using System;
    using System.IO;
    using System.Runtime.Serialization.Formatters.Binary;

    namespace CloneDemo
    {
    [Serializable]
    class DemoClass
    {
    public int i = 0;
    public int[] iArr = { 1, 2, 3 };

    public DemoClass Clone1() //浅clone
    {
    return this.MemberwiseClone() as DemoClass;
    }

    public DemoClass Clone2() //深clone {
    MemoryStream stream = new MemoryStream();
    BinaryFormatter formatter = new BinaryFormatter();
    formatter.Serialize(stream, this);
    stream.Position = 0;
    return formatter.Deserialize(stream) as DemoClass;
    }
    }

    class Program
    {
    static void Main(string[] args)
    {
    DemoClass a = new DemoClass();
    a.i = 10;
    a.iArr = new int[] { 8, 9, 10 };
    DemoClass b = a.Clone1();
    DemoClass c = a.Clone2();

    // 更改 a 对象的iArr[0], 导致 b 对象的iArr[0] 也发生了变化 而 c不会变化
    a.iArr[0] = 88;

    Console.WriteLine("MemberwiseClone");
    Console.WriteLine(b.i);
    foreach (var item in b.iArr)
    {
    Console.WriteLine(item);
    }

    Console.WriteLine("Clone2");
    Console.WriteLine(c.i);
    foreach (var item in c.iArr)
    {
    Console.WriteLine(item);
    }

    Console.ReadLine();
    }
    }
    }


    程序运行结果:
    MemberwiseClone
    10
    88
    9
    10
    Clone2
    10
    8
    9
    10
     
  • 相关阅读:
    题解——loj6279 数列分块入门3 (分块)
    题解——loj6278 数列分块入门2 (分块)
    题解——loj6277 数列分块入门1(分块)
    题解——P1133 教主的花园DP
    题解——P1108低价购买(DP)
    题解——UVA11997 K Smallest Sums
    题解——洛谷P1550 [USACO08OCT]打井Watering Hole(最小生成树,建图)
    题解——CodeForces 438D The Child and Sequence
    题解——ATCoder AtCoder Grand Contest 017 B
    题解——洛谷P2827 NOIP提高组 2016 蚯蚓
  • 原文地址:https://www.cnblogs.com/danshui/p/2286726.html
Copyright © 2011-2022 走看看