为对象创建副本的技术称为拷贝(也叫克隆),拷贝分为浅拷贝和深拷贝。
1.浅拷贝
浅拷贝将对象中的所有字段复制到新的对象(副本)中。值类型字段的值被复制到副本中后,在副本中的修改不会影响到源对象对应的值。而引用类型的字段被复制到副本中的是引用类型的引用,而不是引用的对象,在副本中对引用类型的字段值做修改会影响到源对象本身。
示例代码片:
using System;
namespace _014_ICloneable
{
class Person : ICloneable
{
public int ID { get; set; }
public int Age { get; set; }
public string Name { get; set; }
public Wallet Wallet { get; set; }
#region ICloneable
public object Clone()
{
return this.MemberwiseClone();
}
#endregion
}
class Wallet
{
public int Money { get; set; }
public Wallet(int money)
{
Money = money;
}
public override string ToString()
{
return Money.ToString();
}
}
class Program
{
static void Main(string[] args)
{
{
Person boos = new Person() { ID = 1, Age = 20, Name = "老板1", Wallet = new Wallet(10000) };
Console.WriteLine("---------浅拷贝:更改值前---------");
Person employee = boos.ShallowClone();
Console.WriteLine("employee.ID:" + employee.ID);
Console.WriteLine("employee.Age:" + employee.Age);
Console.WriteLine("employee.Name:" + employee.Name);
Console.WriteLine("employee.wallet:" + employee.Wallet);
Console.WriteLine("---------浅拷贝:更改值后---------");
boos.ID = 2;
boos.Age = 30;
boos.Name = "老板2";
boos.Wallet.Money = 100;
Console.WriteLine("employee.ID:" + employee.ID);
Console.WriteLine("employee.Age:" + employee.Age);
Console.WriteLine("employee.Name:" + employee.Name);
Console.WriteLine("employee.wallet:" + employee.Wallet);
Console.ReadKey();
}
}
}
}
1、注意:Person.Name 属性是 string 类型。
理论上 string 类型是引用类型,但是由于该引用类型的特殊性(无论是实现还是语义)
,Object.MemberwiseClone() 方法仍旧为其创建了副本。 也就是说,在浅拷贝过程,应该将字符串看成是值类型。
2、Person 的 Wallet 属性是一个引用类型,所以,如果改变了源对象 boos 中的值,副本 employee 中的值也会随之一起变动。
3、总结:Object.MemberwiseClone()实现浅拷贝,无法对引用对象深拷贝(字符串比较特殊除外)
2.深拷贝
深拷贝同样,将对象中的所有字段复制到新的对象中。无论是对象的值类型字段,还是引用类型字段,都会被重新创建并赋值,对于副本的修改,不会影响到源对象本身。
深拷贝有很多种实现方法,最简单的方法是手动对字段逐个进行赋值。但这种方法容易出错,如果类型的字段发生变化或有增减,那么该拷贝方法也要发生相应的变化,所以, 建议使用序列化的形式来进行深拷贝。
示例代码片:
class Program
{
static void Main(string[] args)
{
{
Person boos = new Person() { ID = 10, Age = 30, Name = "老板3", Wallet = new Wallet(40000) };
Console.WriteLine("---------深拷贝:更改值前---------");
Person employee = boos.DeepClone();
Console.WriteLine("employee.ID:" + employee.ID);
Console.WriteLine("employee.Age:" + employee.Age);
Console.WriteLine("employee.Name:" + employee.Name);
Console.WriteLine("employee.wallet:" + employee.Wallet);
Console.WriteLine("---------深拷贝:更改值后---------");
boos.ID = 20;
boos.Age = 40;
boos.Name = "老板4";
boos.Wallet.Money = 400;
Console.WriteLine("employee.ID:" + employee.ID);
Console.WriteLine("employee.Age:" + employee.Age);
Console.WriteLine("employee.Name:" + employee.Name);
Console.WriteLine("employee.wallet:" + employee.Wallet);
Console.ReadKey();
}
}
}