前言
有时候我们想克隆一个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]
特性