掘金 后端 ( ) • 2024-04-16 18:01

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>
/// <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;
}

总结

浅拷贝通常用于数据传输,因为它是快速且有效的。但是,如果需要避免意外修改原始对象,则应使用深拷贝。

以下是一些有关何时使用浅拷贝和深拷贝的准则:

  • 使用浅拷贝:
    • 当需要快速创建对象副本时
    • 当原始对象不可变时
    • 当原始对象和副本不会同时使用时
  • 使用深拷贝:
    • 当需要避免意外修改原始对象时
    • 当原始对象和副本需要同时使用时
    • 当原始对象包含引用类型成员时

引用

本文由mdnice多平台发布