c# 引用对象赋值时 为什么需要深度克隆

 更新时间:2020年8月17日 09:12  点击:1435

有基础的开发者都应该很明白,对象是一个引用类型,例如:

object b=new object();
object a=b;

那么a指向的是b的地址,这样在有些时候就会造成如果修改a的值,那么b的值也会跟随着改变(a和b是同一个引用内存地址)。

举一个简单的栗子

public class Item
{
	public string Name { get; set; }
	public string Code { get; set; }
}

private void button1_Click(object sender, EventArgs e)
{
	List<Item> list = new List<Item>();
	Item item = new Item();
	item.Name = "1";
	item.Code = "aaa";
	list.Add(item);
	item = new Item();
	item.Name = "2";
	item.Code = "bbb";
	list.Add(item);
	item = new Item();
	item.Name = "3";
	item.Code = "ccc";
	list.Add(item);


	foreach (var st in list)
	{
		var newItem = st;
		newItem.Code = "ddd";
	}
}


当我们执行到

newItem.Code = "ddd";

的时候,原来的list里的第一项的code也从"aaa"变成了"ddd"


为什么会这样?这里就牵涉到 值类型引用类型


值类型变量的赋值: 值类型变量中保存的是实际数据,在赋值的时候只是把数据复制一份,然后赋给另一个变量。

例子1:

int var1=2;
int var2=var1;  //编译器会先复制var1的值,然后把它赋给var2.很明显var2的值也为2


引用类型变量的赋值:引用类型变量中保存的是“指向实际数据的引用指针”。在进行赋值操作的时候,它和值类型一样,也是先有一个复制的操作,不过它复制的不是实际的数据,而是引用(真实数据的内存地址)。所以引用类型的变量在赋值的时候,赋给另一变量的实际上是内存地址。这样赋值完成后,2个引用变量中保存的是同一引用,他们的指向完全一样。

class MyClass            
{
    public int val;
}
struct MyStruct
{
    public int val;
}
class Program
{
	static void Main(string[] args)
	{
		MyClass objectA=new MyClass();
		MyClass objectB=objectA;                 //引用变量的赋值 赋值操作完成后,两个变量都指向同一内存地址
		objectA.val=10;                               //给objectA.val赋值=10 由于objectB和objectA指向同一内存地址,所以ojbectB.val的值也为10
		objectB.val=20;                               //给objectB.val赋值=20 由于objectB和objectA指向同一内存地址,所以objectA.val的值也为20

		MyStruct structA=new MyStruct();
		MyStruct structB=structA;                //结构是值类型 赋值操作完成后,两个结构中的结构信息一致。注意是“结构中的信息”一致。
		structA.val=30;
		structB.val=40;

		Console.WriteLine(objectA.val);         //输出结果是20
		Console.WriteLine(objectB.val);         //输出结果是20
		Console.WriteLine(structA.val);         //输出结果是30
		Console.WriteLine(structB.val);         //输出结果是40
		Console.ReadLine();
   }
}  

struct结构是值类型


可以看出,值类型变量的赋值操作,仅仅是2个实际数据之间的复制。而引用类型变量的赋值操作,复制的是引用,即内存地址,由于赋值后二者都指向同一内存地址,所以改变其中一个,另一个也会跟着改变,二者就像绑定在了一起。


我们想要a和b都是各自互不影响的,那么只能是完全地新建一个新的对象,并且把现有对象的每个属性的值赋给新的对象的属性。也就是值类型的复制,这个操作就叫深度克隆。


我们可以利用序列化进行对象拷贝,要求对象是序列化的

public static T Clone<T>(T item)
	where T : class
{
	T result = default(T);
	if (null != item)
	{
		MemoryStream ms = new MemoryStream();
		BinaryFormatter bf = new BinaryFormatter();
		bf.Serialize(ms, item);
		ms.Seek(0, SeekOrigin.Begin);
		result = bf.Deserialize(ms) as T;//网上抄的代码总是把这句写在最后,奇葩的大家都是一顿copy
	}
	return result;
}

重新来看文章最开始的示例代码

[Serializable] //为了可以clone,这里需要把类设置可以序列化
public class Item
{
	public string Name { get; set; }
	public string Code { get; set; }
}

private void button1_Click(object sender, EventArgs e)
{
	List<Item> list = new List<Item>();
	Item item = new Item();
	item.Name = "1";
	item.Code = "aaa";
	list.Add(item);
	item = new Item();
	item.Name = "2";
	item.Code = "bbb";
	list.Add(item);
	item = new Item();
	item.Name = "3";
	item.Code = "ccc";
	list.Add(item);


	foreach (var st in list)
	{
		//var newItem = st;
		//newItem.Code = "ddd";

		var newItem = Clone(st);
		newItem.Code = "ddd";
	}
}

此时再运行到

newItem.Code = "ddd";

原来的list的值就不会发生改变了


Over

[!--infotagslink--]

相关文章

  • 深入理解PHP变量的值类型和引用类型

    在PHP中,大部分变量类型,如字符串,整型,浮点,数组等都是值类型的,而类和对象是引用类型,在使用的时候,需要注意这一点。看到网友在讨论PHP的&符号,要彻底理解它的用法,就有必要讨论一下变量的两种形式。PHP的变量在内存中是这样...2015-10-23
  • C#引用类型和值类型的介绍与实例

    这篇文章主要介绍了C#引用类型和值类型,有需要的朋友可以参考一下...2020-06-25
  • JavaScript中各种引用类型的常用操作方法小结

    这篇文章主要介绍了JavaScript中各种引用类型的常用操作方法小结,基本上都用实际代码进行展示,是整理得比较全面的学习笔记,需要的朋友可以参考下...2016-05-09
  • C#引用类型和值类型的适用场合和区别

    今天小编就为大家分享一篇关于C#引用类型和值类型的适用场合和区别,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧...2020-06-25
  • c#值类型和引用类型使用示例

    这篇文章主要介绍了c#值类型和引用类型使用示例,需要的朋友可以参考下...2020-06-25
  • C语言中字符串与各数值类型之间的转换方法

    这篇文章主要介绍了C语言中字符串与各数值类型之间的转换方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2021-03-23
  • c#中值类型和引用类型的基础教程

    这篇文章主要给大家介绍了关于c#中值类型和引用类型的基础教程,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧...2020-06-25
  • C#值类型、引用类型中的Equals和==的区别浅析

    这篇文章主要介绍了C#值类型、引用类型中的Equals和==的区别浅析,本文分别对C#值类型和引用类型中的Equals和==做了讲解和给出了实例,需要的朋友可以参考下...2020-06-25
  • c# 引用类型构造器

    引用类型构造器,是将类型的实例初始化为良好状态的一种特殊方法...2020-06-25
  • 四种引用类型在JAVA Springboot中的使用详解

    这篇文章主要介绍了springboot的四种引用类型,本文通过示例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下...2021-09-03
  • c# 值类型实例构造器

    CLR总是允许创建值类型的实例。另外值类型不一定需要定义构造器,c#编译器不会为值类型生成默认的无参构造器...2020-06-25
  • C#中值类型和引用类型的区别深度分析

    这篇文章主要介绍了C#中值类型和引用类型的区别深度分析,以通俗易懂的语言形象化的分析了C#中值类型和引用类型的区别,对于深入理解C#数据类型有着不错的参考借鉴价值,需要的朋友可以参考下...2020-06-25
  • C#值类型和引用类型的深入理解

    本篇文章主要是对C#中值类型和引用类型进行了详细的分析介绍,需要的朋友可以过来参考下,希望对大家有所帮助...2020-06-25
  • c#字符串值类型与引用类型比较示例

    这篇文章主要介绍了c#字符串值类型与引用类型比较示例,需要的朋友可以参考下...2020-06-25
  • C#中值类型和引用类型解析

    这篇文章主要为大家详细介绍了C#中值类型和引用类型的相关资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...2020-06-25
  • 浅析C# 中的类型系统(值类型和引用类型)

    这篇文章主要介绍了浅析C# 中的类型系统(值类型和引用类型),需要的朋友可以参考下...2020-06-25
  • C#预定义数据类型之值类型和引用类型介绍

    这篇文章主要介绍了C#预定义数据类型之值类型和引用类型介绍,本文着重讲解了引用类型中的object(对象)类型和string(字符串)类型,需要的朋友可以参考下...2020-06-25
  • C# List引用类型克隆的3种方法

    这篇文章主要给大家介绍了关于C# List引用类型克隆的3种方法,包括反射、序列化(依赖Newtonsoft.Json)以及序列化(BinaryFormatter)的实现方法,需要的朋友可以参考借鉴,下面来一起看看吧...2020-06-25
  • c# 引用类型和值类型

    CLR支持两种类型:引用类型和值类型。引用类型总是从托管堆上分配的...2020-06-25
  • c#基础系列之值类型和引用类型的深入理解

    这篇文章主要给大家介绍了关于c#基础系列之值类型和引用类型的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2020-06-25