C# List的深复制

1、关于深拷贝和浅拷贝

C#支持两种类型:值类型和引用类型

值类型(Value Type):如 char, int, float,枚举类型和结构类型

引用类型(Reference Type):如Class类型、接口类型、委托类型、数组类型。

如何来划分它们?

以它们在计算机内存中如何分配来划分。

堆栈(Stack)是一种先进先出的数据结构,在内存中,变量会被分配在堆栈上来进行操作。

堆(Heap)是用于为类型实例(对象)分配空间的内存区域,在堆上创建一个对象,会将对象的地址传递给堆栈上的变量(反过来叫变量指向此对象,或者变量引用此对象)。

值类型与引用类型的区别?

(1)、值类型的变量直接包含其数据

(2)、引用类型的变量则存储对象引用

对于引用类型,两个变量可能引用同一个对象,因此对一个变量的操作可能影响到另一个变量所引用的对象。

对于值类型,每个变量都有自己的数据副本,对于一个变量的操作不可能影响到另一个变量。

值类型隐式继承自System.ValueType,所以不能显示让一个结构继承一个类,C#不支持多继承。 

关于对象克隆的所设计到知识点

浅拷贝:是指将对象中的所有字段逐字复制到一个新对象

对值类型字段只是简单的拷贝一个副本到目标对象,改变目标对象中值类型字段的值不会反映到原始对象中,因为拷贝的是副本。实现浅拷贝需要使用Object类的MemberwiseClose方法用于创建一个浅表副本。

对引用类型字段则是指拷贝他的一个引用到目标对象,改变目标对象中引用类型字段的值会将它反映到原始对象中,因为拷贝的是指向堆是上的一个地址。

深拷贝:深拷贝与浅拷贝不同的是对于引用字段的处理,深拷贝会将在新对象中创建一个新的对象和原始对象字段相同(内容相同)的字段,也就是说这个引用和原始对象引用是不同,我们改变新对象这中这个字段的时候是不会影响到原始对象中对应字段的内容。须实现ICloneable接口中的Clone方法,且需要将被克隆对象加上[Serializable]特性。

引用类型的赋值、浅拷贝和深拷贝的区别

赋值:就是将原来对象的地址给新的对象拷贝一下即可。

浅拷贝:需要继承ICloneable接口(支持克隆,即用现有实例相同的值创建类的新实例),然后用MemberwiseClone方法(创建当前System.Object的浅表对象)。但是需要注意的是MemberwiseClone拷贝方式,首先它是浅拷贝,方法是将所有的值类型字段拷贝一个副本,但是引用类型不会创建副本,仅仅是传递一个相同的地址给新对象,并且新对象和原对象指向的地址是一致的,这里有个问题就是string类型,但是实际上却是和值类型表现一致。

深拷贝:深拷贝与浅拷贝不同的是对于引用字段的处理,深拷贝会将在新对象中创建一个新的对象和原始对象字段相同(内容相同)的字段,也就是说这个引用和原始对象引用是不同,我们改变新对象这中这个字段的时候是不会影响到原始对象中对应字段的内容。须实现ICloneable接口中的Clone方法,且需要将被克隆对象加上[Serializable]特性。

2 List 的深拷贝

代码如下:

  1. /// <summary>
  2. /// Class Program
  3. /// </summary>
  4. class Program
  5. {
  6. /// <summary>
  7. /// Defines the entry point of the application.
  8. /// </summary>
  9. /// <param name="args">The args.</param>
  10. static void Main(string[] args)
  11. {
  12. List<Person> originalList = new List<Person>();
  13. Person person = new Person();
  14. person.Name = "zhangsan";
  15. person.Age = 90;
  16. Job job = new Job();
  17. job.JobName = "开发工程师";
  18. person.PersonJob = job;
  19. originalList.Add(person);
  20.  
  21. person = new Person();
  22. person.Name = "lisi2";
  23. job = new Job();
  24. job.JobName = "开发工程师";
  25. person.PersonJob = job;
  26. originalList.Add(person);
  27.  
  28. person = new Person();
  29. person.Name = "3";
  30. job = new Job();
  31. job.JobName = "测试工程师";
  32. person.PersonJob = job;
  33. originalList.Add(person);
  34.  
  35. Console.WriteLine("原数据如下:");
  36. for (int i = 0; i < originalList.Count; i++)
  37. {
  38. Console.WriteLine("Name:" + originalList[i].Name + ",JobName:" + originalList[i].PersonJob.JobName);
  39. }
  40.  
  41. Console.WriteLine("--------------List深复制------------------");
  42. List<Person> deepCopyList = Clone<Person>(originalList);
  43. deepCopyList[1].Name = "lisi";
  44. for (int i = 0; i < originalList.Count; i++)
  45. {
  46. Console.WriteLine("原数据:Name:" + originalList[i].Name + ",JobName:" + originalList[i].PersonJob.JobName);
  47. Console.WriteLine("深复制后:Name:" + deepCopyList[i].Name + ",JobName:" + deepCopyList[i].PersonJob.JobName);
  48. }
  49.  
  50. Console.WriteLine("----------------List赋值----------------");
  51. List<Person> shallowCopyList = originalList;
  52. shallowCopyList[2].Name = "amy";
  53. shallowCopyList[2].PersonJob.JobName = "产品工程师";
  54. Job modifyJob = new Job();
  55. modifyJob.JobName = "UNKNOWN";
  56. shallowCopyList[1].PersonJob = modifyJob;
  57. for (int i = 0; i < originalList.Count; i++)
  58. {
  59. Console.WriteLine("原数据:Name:" + originalList[i].Name + ",JobName:" + originalList[i].PersonJob.JobName);
  60. Console.WriteLine("浅复制:Name:" + shallowCopyList[i].Name + ",JobName:" + shallowCopyList[i].PersonJob.JobName);
  61. }
  62.  
  63. Console.Read();
  64. }
  65.  
  66. /// <summary>
  67. /// Clones the specified list.
  68. /// </summary>
  69. /// <typeparam name="T"></typeparam>
  70. /// <param name="List">The list.</param>
  71. /// <returns>List{``0}.</returns>
  72. public static List<T> Clone<T>(object List)
  73. {
  74. using (Stream objectStream = new MemoryStream())
  75. {
  76. IFormatter formatter = new BinaryFormatter();
  77. formatter.Serialize(objectStream, List);
  78. objectStream.Seek(0, SeekOrigin.Begin);
  79. return formatter.Deserialize(objectStream) as List<T>;
  80. }
  81. }
  82. }

两个测试类代码如下:

  1. /// <summary>
  2. /// Class Person
  3. /// </summary>
  4. [Serializable]
  5. public class Person
  6. {
  7. /// <summary>
  8. /// Gets or sets the person job.
  9. /// </summary>
  10. /// <value>The person job.</value>
  11. public Job PersonJob
  12. {
  13. get;
  14. set;
  15. }
  16.  
  17. /// <summary>
  18. /// Gets or sets the name.
  19. /// </summary>
  20. /// <value>The name.</value>
  21. public string Name
  22. {
  23. get;
  24. set;
  25. }
  26.  
  27. /// <summary>
  28. /// Gets or sets the age.
  29. /// </summary>
  30. /// <value>The age.</value>
  31. public int Age
  32. {
  33. get;
  34. set;
  35. }
  36.  
  37. /// <summary>
  38. /// Clones this instance.
  39. /// </summary>
  40. /// <returns>System.Object.</returns>
  41. public object Clone()
  42. {
  43. BinaryFormatter formatter = new BinaryFormatter(null, new System.Runtime.Serialization.StreamingContext(System.Runtime.Serialization.StreamingContextStates.Clone));
  44. MemoryStream stream = new MemoryStream();
  45. formatter.Serialize(stream, this);
  46. stream.Position = 0;
  47. object clonedObj = formatter.Deserialize(stream);
  48. stream.Close();
  49. return clonedObj;
  50. }
  51. }
  52.  
  53. /// <summary>
  54. /// Class Job
  55. /// </summary>
  56. [Serializable]
  57. public class Job
  58. {
  59. /// <summary>
  60. /// Gets or sets the name of the job.
  61. /// </summary>
  62. /// <value>The name of the job.</value>
  63. public string JobName
  64. {
  65. get;
  66. set;
  67. }
  68. }

结果如下:

3 不用加Serializable的深复制方法

  1. public static T DeepCopy<T>(T obj)
  2. {
  3. object retval;
  4. using (MemoryStream ms = new MemoryStream())
  5. {
  6. XmlSerializer xml = new XmlSerializer(typeof(T));
  7. xml.Serialize(ms, obj);
  8. ms.Seek(0, SeekOrigin.Begin);
  9. retval = xml.Deserialize(ms);
  10. ms.Close();
  11. }
  12. return (T)retval;
  13. }

  

作者:BestNow
出处:http://www.cnblogs.com/BestNow/ 
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。 

C# List的深复制(转)的更多相关文章

  1. 学习笔记--C#深复制和浅复制

    参考博客:http://www.cnblogs.com/nliao/archive/2012/11/18/2776114.html 例子网上都有很多,我也就不列了. 其实很久以前就明白了这两者的区别, ...

  2. iOS 浅复制和深复制的深层理解,含示例

    转载:https://www.zybuluo.com/MicroCai/note/50592 版权归 @MicroCai 所有 以下是正文: 浅复制就是指针拷贝:深复制就是内容拷贝. 集合的浅复制 ( ...

  3. PHP浅复制与深复制

    原文链接:http://www.orlion.ga/731/ php用clone复制对象有一个问题,下面用代码来说明问题: class Foo{ public $bar; public $name; ...

  4. .Net深复制、浅复制

    在.Net,大家都知道引用类型的深复制.浅复制吧. ,一般int等值类型是值类型(复制时是直接传值),一般的类(List<T>,Class)是引用类型(复制时传地址),默认是浅复制.若ob ...

  5. python中深复制与浅复制

    转载:http://blog.csdn.net/vicken520/article/details/8227524 java中也经常碰见这种问题.时间原因就不写java方面啦 Python深复制浅复制 ...

  6. javascript中关于深复制与浅复制的问题

    在javascript中,变量的类型分为基本类型和引用类型. 对于基本类型的变量来说,值的复制以及作为函数参数实参传递的过程都是值的复制传递,换句话说,是会在内存中开辟出一个新空间用于存放新的值的.这 ...

  7. js 对象深复制,创建对象和继承

    js 对象深复制,创建对象和继承.主要参考高级编程第三版,总结网上部分资料和自己的代码测试心得.每走一小步,就做一个小结. 1.对象/数组深复制 一般的=号传递的都是对象/数组的引用,如在控制台输入 ...

  8. C++学习基础七——深复制与浅复制

    一.深复制与浅复制基本知识 深复制和浅复制,又称为深拷贝和浅拷贝. 深复制和浅复制的区别如下图1所示: 图1 图1表示的是,定义一个类CDemo,包含int a和char *str两个成员变量, 当深 ...

  9. javascript中的浅复制和深复制

    //浅复制:实现基本类型的复制没有问题,但是复制的是引用类型的话,则修改child将会修改parent function extend(parent,child){ var child = child ...

  10. Java-clone浅/深复制

    Object中的clone方法为复制当前对象 protected native Object clone() throws CloneNotSupportedException; 想要使用这个方法需要 ...

随机推荐

  1. EchoService

    dubbo为consumer端的代理对象实现了EchoService接口. 使用示例: <dubbo:reference id="hello" interface=" ...

  2. SQL触发器实例(上)

    --1.) 创建测试用的表(testTable) if exists (select * from sysobjects where name='testTable') drop table test ...

  3. Openwrt VLAN Configure(2)

    1      Scope of Document This document describes vlan design on nodewrt2p 2      Requiremen 2.1     ...

  4. asp.net文件压缩,下载,物理路径,相对路径,删除文件

    知识动手实践一次,就可以变成自己的了.不然一直是老师的,书本的. 这几天做了一个小小的项目,需要用到文件下载功能,期初想到只是单个的文件,后面想到如果很多文件怎么办?于是又想到文件压缩.几经波折实践, ...

  5. kettle使用命令行来运行ktr和kjb

    1:cmd方式运行 1.ktr的运行:运行transformation文件是通过Pan.bat来运行的. 打开cmd命令行窗口,转到Pan.bat所在的目录,如d:\data-integration, ...

  6. FE英文缩写

    PM:product manager,产品经理,又称品牌经理,brand managerRD:research and development enginner,研发工程师FE:front-end d ...

  7. Criteria 使用指南

    转自:http://www.blogjava.net/jerry-zhaoj/archive/2009/03/03/257546.html Restrictions的几个常用限定查询方法如下表所示: ...

  8. Java——线程定时器

    body, table{font-family: 微软雅黑; font-size: 10pt} table{border-collapse: collapse; border: solid gray; ...

  9. postgresql的copy

    https://www.postgresql.org/docs/current/static/sql-copy.html 一.Copy的基本语法 Copy的作用是复制数据在数据表和文件之间. Copy ...

  10. C# winform 编程 向ACCESS数据库导入EXCEL表使用心得

    public string MyConnectionString = "Provider=Microsoft.Jet.OLEDB.4.0;Data Source=ErLake.mdb&quo ...