C#中的对象深拷贝和浅拷贝
概述
在C#中,对象拷贝是指将一个对象的副本创建到另一个对象中。对象拷贝通常用于数据传输或创建对象的新实例。
C#中有两种主要的拷贝方式:浅拷贝和深拷贝
1. 浅拷贝
浅拷贝是指只拷贝对象的值类型成员,而引用类型成员的引用则保持不变。这意味着新的对象和原始对象将共享所有引用类型成员的实际对象。
实现方式
- this.MemberwiseClone();
示例代码
实体
public class Person
{
public Person()
{
this.Address = new Address();
}
public string Name { get; set; }
public int Age { get; set; }
public Address Address { get; set; }
public Person Clone()
{
return (Person)this.MemberwiseClone();
}
}
public class Address
{
public string Street { get; set; }
public string City { get; set; }
}
调用
Person person1 = new Person()
{
Name = "张三",
Address = new Address()
{
City = "北京",
}
};
Person person2 = person1.Clone();//浅拷贝
//修改原对象的属性
person1.Address.City = "上海";
//修改副本对象的属性
person2.Name = "李四";
person2.Address.City = "昆明";
string result = $"原对象{JsonConvert.SerializeObject(person1)}。副本{JsonConvert.SerializeObject(person2)}";
MessageBox.Show($"浅拷贝:原对象和副本修改引用类型属性后相互影响。{result}");
2. 深拷贝
深拷贝是指不仅拷贝对象的值类型成员,而且还拷贝所有引用类型成员的实际对象。这意味着新的对象将拥有其引用类型成员的完全独立副本。
实现方式
- 反射
- 序列化
- 对象映射(三方开源如TinyMapper、AutoMapper)。
示例代码
/// <summary>
/// 深拷贝
/// </summary>
public static void Copy2()
{
Person person1 = new Person()
{
Name = "张三",
Address = new Address()
{
City = "北京",
}
};
//Person person2 = CreateDeepCopy(person1);//深拷贝1反射
// Person person2 =JsonConvert.DeserializeObject<Person>(JsonConvert.SerializeObject(person1));//深拷贝2序列化
Person person2 =person1.MapTo<Person,Person>();//深拷贝3对象映射
//修改原对象的属性
person1.Address.City = "上海";
//修改副本对象的属性
person2.Name = "李四";
person2.Address.City = "昆明";
string result = $"原对象{JsonConvert.SerializeObject(person1)}。副本{JsonConvert.SerializeObject(person2)}";
MessageBox.Show($"深拷贝:原对象和副本不相互影响。{result}");
}
/// <summary>
/// 使用反射进行深拷贝
/// </summary&ggt;
/// <typeparam name="T"></typeparam>
/// <param name="original"></param>
/// <returns></returns>
static T CreateDeepCopy<T>(T original)
{
if (original == null)
{
return default(T);
}
Type type = original.GetType();
object newObject = Activator.CreateInstance(type);
foreach (FieldInfo fieldInfo in type.GetFields())
{
if (fieldInfo.IsStatic)
{
continue;
}
object value = fieldInfo.GetValue(original);
fieldInfo.SetValue(newObject, CreateDeepCopy(value));
}
return (T)newObject;
}
总结
浅拷贝通常用于数据传输,因为它是快速且有效的。但是,如果需要避免意外修改原始对象,则应使用深拷贝。
以下是一些有关何时使用浅拷贝和深拷贝的准则:
- 使用浅拷贝:
- 当需要快速创建对象副本时
- 当原始对象不可变时
- 当原始对象和副本不会同时使用时
- 使用深拷贝:
- 当需要避免意外修改原始对象时
- 当原始对象和副本需要同时使用时
- 当原始对象包含引用类型成员时