C# List引用类型克隆的3种方法

本文介绍了在C#中如何克隆一个List<T>,提供了三种实现方式:1) 使用反射创建对象并复制属性;2) 利用Newtonsoft.Json进行序列化和反序列化;3) 通过BinaryFormatter进行序列化和反序列化。每种方法都有其适用场景,注意反射不需要额外依赖,而序列化可能需要引入外部库。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

前言

有时候我们想克隆一个List去做别的事,而不影响原来的List,我们直接在list后面加上小点点,发现并没有Clone这样的扩展函数。这时候就只有自己扩展了。

尝试了三种方式,测试都通过了,至于性能方面我还没有做测试。

下面话不多说了,来一起看看详细的介绍吧

一、反射

public static List<T> Clone<T>(this List<T> list) where T : new()
  {
   List<T> items = new List<T>();
   foreach (var m in list)
   {
    var model = new T();
    var ps = model.GetType().GetProperties();
    var properties = m.GetType().GetProperties();
    foreach (var p in properties)
    {
     foreach (var pm in ps)
     {
      if (pm.Name == p.Name)
      {
       pm.SetValue(model, p.GetValue(m));
      }
     }
    }
    items.Add(model);
   }
   return items;
  }

二、序列化(依赖Newtonsoft.Json)

public static List<T> Clone<T>(this List<T> list) where T : new()
{
var str = JsonConvert.SerializeObject(list);
return JsonConvert.DeserializeObject<List<T>>(str);
}

三、序列化(BinaryFormatter

public static List<T> Clone<T>(this List<T> list)
  {
   using (Stream objectStream = new MemoryStream())
   {
    IFormatter formatter = new BinaryFormatter();
    formatter.Serialize(objectStream, list);
    objectStream.Seek(0, SeekOrigin.Begin);
    return (List<T>)formatter.Deserialize(objectStream);
   }
  }

测试

private void Test()
{
 List<NormalSetting> list = new List<NormalSetting>();
 list.Add(new NormalSetting { RedisIp = "123" });
 List<NormalSetting> items = list.Clone();
 list[0].RedisIp = "456";
 logMessager.Show("{0}:{1}", list[0].RedisIp, items[0].RedisIp);
}

在这里插入图片描述

注意事项:

第一种方式无需任何依赖。

第二种方式需要Newtonsoft.Json,如果项目中没有用到它,不推荐使用这种方式。

第三种方式序要给引用类型实体加上[Serializable]特性

C#中,深度克隆一个对象通常涉及到复杂的数据结构,因为默认的赋值操作(=``)只会复制引用,而不是数据本身。对于基本类型、值类型以及非集合类,可以简单地通过复制构造函数或赋值操作来实现浅拷贝。但对于复杂对象(如包含嵌套对象或集合的对象),你需要递归处理,以确保每个引用都被替换为其新实例。 以下是深度克隆一个自定义类及其嵌套对象的一个通用方法: ```csharp public class MyClass { public object NestedObject { get; set; } // ...其他属性... } MyClass CloneObject(MyClass original) { var clone = new MyClass(); clone.NestedObject = original.NestedObject == null ? null : (original.NestedObject is MyNestedClass ? new MyNestedClass((MyNestedClass)original.NestedObject) : original.NestedObject); // 根据类型深复制 // 对于所有嵌套属性也做同样的处理 // ... return clone; } ``` 如果嵌套的是复杂的数据结构,比如列表、字典或其他容器类型,你也需要遍历它们并创建新的实例: ```csharp List<MyNestedClass> originalList = original.MyList; clone.MyList = new List<MyNestedClass>(originalList.Count); foreach (var item in originalList) { clone.MyList.Add(item.Clone()); // 如果NestedObject类也有Clone()方法 } ``` 如果你有一个集合类,那么可以使用序列化和反序列化的方式来进行深度克隆: ```csharp using System.IO; byte[] serialized = Serializers.Serialize(original); MyClass clonedObject = Serializers.Deserialize<MyClass>(serialized); ``` 这里假设`Serializers`是一个辅助类库,用于序列化和反序列化。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值