C# 理解浅拷贝和深拷贝(shallow copy VS deep copy )(1)

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








һϢϵͳܱǸύлʵкϵͳĹؼʻIPܵ˷ƣβЧϵͳѼ¼IPύݡע⣬ҪύκΥҹ涨ݣصϢΪgooglesyndication
一对象。代码实现如下:

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 既可作为深层副本实现,也可作为浅表副本实现。在深层副本中,所有的对象都是重复的;而在浅表副本中,只有顶级对象是重复的,并且顶级以下的对象包含引用。 结果克隆必须与原始实例具有相同的类型或是原始实例的兼容类型。

代码实现如下:

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 相同的结构(表架构、关系和约束)和数据。注意要是已树立这些类的子类,则副本也将属于相同的子类。

好像既不是浅拷贝,又 һϢϵͳ









һϢϵͳܱǸύлʵкϵͳĹؼʻIPܵ˷ƣβЧϵͳѼ¼IPύݡע⣬ҪύκΥҹ涨ݣصϢΪgooglesyndication
һϢϵͳ








һϢϵͳܱǸύлʵкϵͳĹؼʻIPܵ˷ƣβЧϵͳѼ¼IPύݡע⣬ҪύκΥҹ涨ݣصϢΪgooglesyndication
cle.aspx?id=652200 target='_blank'>Flash在JavaScript应用程序的交互
  • 加速Win2003关机速度
  • 利用反向访问控制列表管理网络和防范病毒
  • MySql服务器集群的可行性方案进行详细的分析和比较
  • php代码:用Smarty批量生成html