C#对象的浅拷贝,深拷贝及利用序列化等多种方式实现深拷贝

C#中有两种类型变量,一种 是值类型变量,一种是引用类型变量,对于值类型变量,深拷贝和前拷贝都是通过赋值操作符号(=)实现,其效果一致,将对象中的值类型的字段拷贝到新的对象中.这个很容易理解。 本文重点讨论引用类型变量的拷贝机制和实现。

  C#中引用类型对象的copy操作有两种:

  浅拷贝(影子克隆/shallow copy):只复制对象的值类型字段,对象的引用类型,仍属于原来的引用. 深拷贝(深度克隆):不仅复制对象的值类型字段,同时也复制原对象中的对象.就是说完全是新对象产生的.

  浅拷贝和深拷贝之间的区别:浅拷贝是指将对象中的数值类型的字段拷贝到新的对象中,而对象中的引用型字段则指复制它的一个引用到目标对象。

  注意:string类型有点特殊,对于浅拷贝,类值类型对象进行处理。

  浅拷贝的实现

  使用Object类MemberwiseClone实现

  MemberwiseClone:创建当前 Object 的浅表副本。

  MemberwiseClone 方法创建一个浅表副本,方法是创建一个新对象,然后将当前对象的非静态字段复制到该新对象。如果字段是值类型的,则对该字段执行逐位复制。如果字段是引用类型,则复制引用但不复制引用的对象;因此,原始对象及其复本引用同一对象。

  代码实现如下:


     public class Person
  {
  public int Age { get; set; }
  public string Address { get; set; }
  public Name Name { get; set; }
  public object Clone()
  {
  return this.MemberwiseClone();
  }
  }
  public class Name
  {
  public Name(string frisName,string lastName)
  {
  FristName = frisName;
  LastName = lastName;
  }
  public string FristName { get; set; }
  public string LastName { get; set; }
  }

 

     赋值操作(=)VS使用Object类MemberwiseClone实现

  对于引用类型的变量,我们有种误解,认为赋值操作就是浅拷贝一种,其实不然,两者有区别。

  浅拷贝(shallow copy)对于引用类型对象中的值类型字段进行了逐位复制。赋值运算符只是把源对象的引用赋值给目的对象,两者引用同一个对象。 浅拷贝后的对象的值类型字段更改不会反映到源对象,而赋值运算后的对象的值类型字段更改会反映到源对象 代码实现如下:

  


     public class Person
  {
  public int Age { get; set; }
  public string Address { get; set; }
  public Name Name { get; set; }
  }
  public class Name
  {
  public Name(string frisName,string lastName)
  {
  FristName = frisName;
  LastName = lastName;
  }
  public string FristName { get; set; }
  public string LastName { get; set; }
  }

 

  深拷贝实现

  相对于浅拷贝,是指依照源对象为原型,创建一个新对象,将当前对象的所有字段进行执行逐位复制并支持递归,不管是是值类型还是引用类型,不管是静态字段还是非静态字段。

  在C#中,我们们有三种方法实现深拷贝

  实现ICloneable接口,自定义拷贝功能。

  ICloneable 接口,支持克隆,即用与现有实例相同的值创建类的新实例。

  ICloneable 接口包含一个成员 Clone,它用于支持除 MemberwiseClone 所提供的克隆之外的克隆。Clone 既可作为深层副本实现,也可作为浅表副本实现。在深层副本中,所有的对象都是重复的;而在浅表副本中,只有顶级对象是重复的,并且顶级以下的对象包含引用。 结果克隆必须与原始实例具有相同的类型或是原始实例的兼容类型。

  代码实现如下:


 Code
  public class Person:ICloneable
  {
  public int Age { get; set; }
  public string Address { get; set; }
  public Name Name { get; set; }
  public object Clone()
  {
  Person tem = new Person();
  tem.Address = this.Address;
  tem.Age = this.Age;
  tem.Name = new Name(this.Name.FristName, this.Name.LastName);
  return tem;
  }
  }
  public class Name
  {
  public Name(string frisName, string lastName)
  {
  FristName = frisName;
  LastName = lastName;
  }
  public string FristName { get; set; }
  public string LastName { get; set; }
  }

 

  大家可以看到,Person类继承了接口ICloneable并手动实现了其Clone方法,这是个简单的类,试想一下,如果你的类有成千上万个引用类型成员(当然太夸张,几十个还是有的),这是不是份很恐怖的劳力活?

  序列化/反序列化类实现

  不知道你有没有注意到DataSet对象,对于他提供的两个方法:

  DataSet.Clone 方法,复制 DataSet 的结构,包括所有 DataTable 架构、关系和约束。不要复制任何数据。

  新 DataSet,其架构与当前 DataSet 的架构相同,但是不包含任何数据。注意 如果已创建这些类的子类,则复本也将属于相同的子类。

  DataSet.Copy 方法复制该 DataSet 的结构和数据.

  新的 DataSet,具有与该 DataSet 相同的结构(表架构、关系和约束)和数据。注意如果已创建这些类的子类,则副本也将属于相同的子类。

  好像既不是浅拷贝,又不是深拷贝,是不是很失望?但是两个结合起来不是我们要的深拷贝吗?看看DataSet的实现,注意序列化接口:ISerializable

  序列化是将对象或对象图形转换为线性字节序列,以存储或传输到另一个位置的过程。

  反序列化是接受存储的信息并利用它重新创建对象的过程。

  通过 ISerializable 接口,类可以执行其自己的序列化行为。

  转换为线性字节序列后并利用其重新创建对象的过程是不是和我们的深拷贝的语意“逐位复制”很相像?

  代码实现如下:


      [Serializable]
  public class Person : ICloneable
  {
  public int Age { get; set; }
  public string Address { get; set; }
  public Name Name { get; set; }
  public object Clone()
  {
  using (MemoryStream ms = new MemoryStream(1000))
  {
  object CloneObject;
  BinaryFormatter bf = new BinaryFormatter(null, new StreamingContext(StreamingContextStates.Clone));
  bf.Serialize(ms, this);
  ms.Seek(0, SeekOrigin.Begin);
  // 反序列化至另一个对象(即创建了一个原对象的深表副本)
  CloneObject = bf.Deserialize(ms);
  // 关闭流
  ms.Close();
  return CloneObject;
  }
  }
  }
  [Serializable]
  public class Name
  {
  public Name(string frisName, string lastName)
  {
  FristName = frisName;
  LastName = lastName;
  }
  public string FristName { get; set; }
  public string LastName { get; set; }
  }
  }

 

  注意:通过序列化和反序列化实现深拷贝,其和其字段类型必须标记为可序列化类型,既添加特性(Attribute)[Serializable]。

  通过反射实现

  通过序列化/反序列化方式我们能比较流畅的实现深拷贝,但是涉及到IO操作,托管的的环境中,IO操作比较消耗资源。 能不能有更优雅的解决方案。CreateInstance,对,利用反射特性。这个方法大家可以参考这篇博客:http://rubenhak.com/?p=70 文章反射类的Attribute,利用Activator.CreateInstance New一个类出来(有点像DataSet.Clone先获得架构),然后利用PropertyInfo的SetValue和GetValue方法,遍历的方式进行值填充。

  代码实现如下:


 public class Person
  {
  private List _friends = new List();
  public string Firstname { get; set; }
  public string Lastname { get; set; }
  [Cloneable(CloneableState.Exclude)]
  [Cloneable(CloneableState.Include, "Friends")]
  public List Friends { get { return _friends; } }
  [Cloneable(CloneableState.Exclude)]
  public PersonManager Manager { get; set; }
  }

 

时间: 2024-09-14 05:03:12

C#对象的浅拷贝,深拷贝及利用序列化等多种方式实现深拷贝的相关文章

C#对象的浅拷贝,深拷贝

C#中有两种类型变量,一种 是值类型变量,一种是引用类型变量,对于值类型变量,深拷贝和前拷贝都是通过赋值操作符号(=)实现,其效果一致,将对象中的值类型的字段拷贝到新的对象中.这个很容易理解. 本文重点讨论引用类型变量的拷贝机制和实现. C#中引用类型对象的copy操作有两种: 浅拷贝(影子克隆/shallow copy):只复制对象的值类型字段,对象的引用类型,仍属于原来的引用. 深拷贝(深度克隆):不仅复制对象的值类型字段,同时也复制原对象中的对象.就是说完全是新对象产生的. 浅拷贝和深拷贝

Java利用序列化实现对象深度clone的方法_java

本文实例讲述了Java利用序列化实现对象深度clone的方法.分享给大家供大家参考.具体实现方法如下: ByteArrayOutputStream byteOut = new ByteArrayOutputStream(); ObjectOutputStream out = new ObjectOutputStream(byteOut); out.writeObject(obj); ByteArrayInputStream byteIn = new ByteArrayInputStream(by

WPS演示教程:利用创建播放器方式在wpp中插入flash

wpp课件在别的电脑上演示,发现课件中插入的Flash动画不能正常播放,一检查竟然未装有Flash播放程序.为避免这种情况发生,可以采用利用创建播放器方式插入flash. 第1步 用Macromedia flash palyer打开要插入的swf文件"葬花",单击"文件"→"创建播放器"命令,在弹出的对话框中,选择好保存位置和文件名称,单击"保存"按钮. 文件变成带播放器的自动执行文件exe格式,可以在没有flash播放器的机

使用fancybox的时候利用传参的方式进行图片显示,如何同时传递多个图片

问题描述 使用fancybox的时候利用传参的方式进行图片显示,如何同时传递多个图片 要实现的效果就是通过点击一个按钮,调用一个JS函数,同时传递多个要显示图片的地址(图片个数不确定),从而实现如fancybox图片集效果. 解决方案 点击按钮的时候你将图片全部添加到dom中,再调用fancybox方法初始化图片效果 DEMO <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.

js对象的浅拷贝和深拷贝

浅拷贝是直接复制源对象的属性和方法. 深拷贝是在浅拷贝的基础上,如果源对象属性也是一个对象,那么连这个对象也复制,也就是在内存中又开辟了一块地方用来存放. 下面是浅拷贝的代码示例: function extend (a, b){ //a为目标的对象名,b为源对象名 for(var i in b){     a[i] = b[i] }     return a; } 深拷贝的代码暂时没写出来,有兴趣的可以去看一下JQ的源码,里面有相应的代码$.extends. 一.数组的深浅拷贝 在使用JavaS

关于序列化:PHP 拥有序列化方法,可以返回对象的字符串表示。但序列化只保存对象的成员数据而不包话方法

对象|数据|字符串     PHP不支持永久对象,在OOP中永久对象是可以在多个应用的引用中保持状态和功能的对象,这意味着拥有将对象保存到一个文件或数据库中的能力,而且可以在以后装入对象.这就是所谓的序列化机制.PHP 拥有序列化方法,它可以通过对象进行调用,序列化方法可以返回对象的字符串表示.然而,序列化只保存了对象的成员数据而不包话方法. 在PHP4中,如果你将对象序列化到字符串$s中,然后释放对象,接着反序列化对象到$obj,你可以继续使用对象的方法!我不建议这样去做,因为(a)文档中没有

C#存取SQL Server数据库之二:利用序列化进行类链表存取(ArrayList,varbina

创建项目 1.      添加一个名为RWTest的表到 SQL Server MYTest 数据库. 表字段设置如下:  a.      唯一标识字段名称为"ID",类型为Int.  b.       名称为"Description"的VarChar类型的字段,字段长度为50.  c.      名称为"Data" 的varbinary(Max) 类型的字段. 2.      启动 Visual Studio .NET, 并创建一个新的 Vis

Json_lib 序列化对象 如果属性为null 不序列化

问题描述 public class Student {private int id;private String name; public int getId() {return id;}public void setId(int id) {this.id = id;}public String getName() {return name;}public void setName(String name) {this.name = name;}}//正常用jsonObject序列化后得到字符串

利用CORS实现POST方式跨域请求数据

CORS全名Cross-Origin Resource Sharing,顾名思义:跨域分享资源,这是W3C制定的跨站资源分享标准. 目前包括IE10+.chrome.safari.FF都提供了XMLHttpRequest对象对该标准的支持,在更老的IE8中则提供了xDomainRequest对象,部分实现了该标准: 下面是创建request对象的代码: var url = "http://www.111cn.net /1.php"; if (XMLHttpRequest) {