最近做專案中,有一個場景需要複製list給其他物件的屬性賦值,然後再去根據物件的其他屬性操作list的元素資料,其實就是一個list的複製問題
程式碼還原
一個list
internal class Program
{
static void Main(string[] args)
{
Console.WriteLine("Hello, World!");
var lista = new List<People>();
lista.Add(new People()
{
Name = "LaoWang",
Age= 1
});
lista.Add(new People()
{
Name = "LaoLi",
Age = 2
});
lista.Add(new People()
{
Name = "LaoZhang",
Age = 3
});
var listb = new List<People>(lista);
listb.Add(new People()
{
Name = "LaoLiu",
Age = 4
});
Console.WriteLine("lista:");
foreach (var item in lista)
{
Console.WriteLine(item.Name);
Console.WriteLine(item.Age);
}
Console.WriteLine(" ");
Console.WriteLine("listb:");
foreach (var item in listb)
{
Console.WriteLine(item.Name);
Console.WriteLine(item.Age);
}
}
}
public record People
{
public string Name { get; set; }
public int Age { get; set; }
}
輸出結果:
Hello, World!
lista:
LaoWang
1
LaoLi
2
LaoZhang
3
listb:
LaoWang
1
LaoLi
2
LaoZhang
3
LaoLiu
4
增加了元素,表面上是沒有什麼影響的,但是如果我修改呢:
foreach (var item in listb)
{
item.Name = "DDl";
item.Age = 8;
}
Console.WriteLine("new lista:");
foreach (var item in lista)
{
Console.WriteLine(item.Name);
Console.WriteLine(item.Age);
}
Console.WriteLine(" ");
Console.WriteLine("new listb:");
foreach (var item in listb)
{
Console.WriteLine(item.Name);
Console.WriteLine(item.Age);
}
結果變成了:
new lista:
DDl 8
DDl 8
DDl 8
new listb:
DDl 8
DDl 8
DDl 8
DDl 8
連帶著以前的一起改了,這可不是我們想要的結果。所以這種複製算淺克隆。
如何連元素一起復制(深克隆)
- 手動複製,foreach 迴圈 new物件手動插入新的list
- 第三方工具 automapper等
- 序列化和反序列化,透過將物件序列化為一個位元組流,然後再反序列化回來,可以實現深克隆
- 表示式樹或者反射,程式碼如下
public static T DeepClone<T>(T obj)
{
using (var ms = new MemoryStream())
{
var formatter = new System.Runtime.Serialization.Formatters.Binary.BinaryFormatter();
formatter.Serialize(ms, obj);
ms.Seek(0, SeekOrigin.Begin);
return (T)formatter.Deserialize(ms);
}
}
public static List<T> DeepCloneList<T>(List<T> list)
{
return list.Select(item => DeepClone(item)).ToList();
}
總結
以前由於經常做類似的,理解還是很清楚,好久不做到了現在一下就踩坑了,所以為了以後不繼續踩坑,決定記錄下來。