前言
有时候我们想克隆一个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#中如何克隆一个List<T>,提供了三种实现方式:1) 使用反射创建对象并复制属性;2) 利用Newtonsoft.Json进行序列化和反序列化;3) 通过BinaryFormatter进行序列化和反序列化。每种方法都有其适用场景,注意反射不需要额外依赖,而序列化可能需要引入外部库。
421

被折叠的 条评论
为什么被折叠?



