zoukankan      html  css  js  c++  java
  • c#实现深拷贝的几种方法

      为什么要用到深拷贝呢?比如我们建了某个类Person,并且实例化出一个对象,然后,突然需要把这个对象复制一遍,并且复制出来的对象要跟之前的一模一样,来看下我们一般会怎么做,看代码

      

    public class Person
    {
      public string Name { get; set; }
    }

    class Program
    {
      static void Main(string[] args)
        {
          Person sourceP = new Person() { Name = "大哥" };
          Person copyP = sourceP;
          copyP.Name = "大哥大"; // 拷贝对象改变Name值
          // 结果都是"大哥",因为实现的是浅拷贝,一个对象的改变都会影响到另一个对象
          Console.WriteLine("Person.Name: [SourceP: {0}] [CopyP:{1}]", sourceP.Name, copyP.Name);
          Console.Read();
        }
    }

    运行结果如图

     

    可以看到虽然复制了一份sourceP对象,但是修改新对象copyP的Name属性时,居然把原来的sourceP对象的值也改了。这里的原理是因为“=”号,在对引用类型使用时,仅仅是新建一个新的引用变量,然后把引用地址复制给了新的引用变量而已,并没有复制真正的内容,这时候如果需要复制真正内容的话,就需要用到深拷贝的方式了。

    几种常见的深拷贝方式

    1、利用反射实现

    public static T DeepCopyByReflection<T>(T obj)
    {
      if (obj is string || obj.GetType().IsValueType)
      return obj;

      object retval = Activator.CreateInstance(obj.GetType());
      FieldInfo[] fields = obj.GetType().GetFields(BindingFlags.Public|BindingFlags.NonPublic|BindingFlags.Static|BindingFlags.Instance);
      foreach(var field in fields)
      {
        try
        {
          field.SetValue(retval, DeepCopyByReflection(field.GetValue(obj)));
        }
        catch { }
      }

      return (T)retval;
    }

    2、利用二进制序列化和反序列化

    public static T DeepCopyByBinary<T>(T obj)
    {
      object retval;
      using (MemoryStream ms = new MemoryStream())
      {
        BinaryFormatter bf = new BinaryFormatter();
        bf.Serialize(ms, obj);
        ms.Seek(0, SeekOrigin.Begin);
        retval = bf.Deserialize(ms);
        ms.Close();
      }
      return (T)retval;
    }

    注意,使用二进制序列化和反序列化时,在需要序列化的类上要加上[Serializable]

    [Serializable]
    public class Person
    {
      public string Name { get; set; }
    }

    3、利用xml序列化和反序列化

    public static T DeepCopyByXml<T>(T obj)
    {
      object retval;
      using (MemoryStream ms = new MemoryStream())
      {
        XmlSerializer xml=new XmlSerializer(typeof(T));
        xml.Serialize(ms, obj);
        ms.Seek(0, SeekOrigin.Begin);
        retval = xml.Deserialize(ms);
        ms.Close();
      }
      return (T)retval;
    }

  • 相关阅读:
    软件工程网络1514龚厦彬个人阅读作业2-提出问题
    软件工程网络15个人阅读作业1 201521123115 龚厦彬
    Java课程设计——学生信息系统(团队)
    Java程序设计——学生基本信息管理系统
    201521123115《Java程序设计》第14周学习总结
    201521123115《Java程序设计》第13周学习总结
    201521123115 《Java程序设计》第12周学习总结
    毕设01--第一阶段的内容
    网络15软工个人作业5——软件工程总结
    个人作业4——alpha阶段个人总结
  • 原文地址:https://www.cnblogs.com/luoocean/p/10441587.html
Copyright © 2011-2022 走看看