为什么列表的副本仍然使用c#改变原始列表的属性?
本文关键字:列表 改变 原始 属性 副本 为什么 | 更新日期: 2023-09-27 18:04:04
假设我有一个class
public class Employee
{
public string FirstName { get; set; }
public string LastName { get; set; }
public bool isActive { get; set; }
}
并像这样使用:
List<Employee> Employees = new List<Employee>();
Employees.Add(new Employee { FirstName = "firstname", LastName = "lastname", isActive = true });
List<Employee> EmployeesCopy = new List<Employee>(Employees);
EmployeesCopy[0].isActive = false;
为什么改变EmployeesCopy
的isActive
属性也会改变原列表中的属性?
因为新列表仍然包含对相同雇员对象的引用。您可以在新列表中创建新列表,如下所示:
List<Employee> Employees = new List<Employee>();
Employees.Add(new Employee { FirstName = "firstname", LastName = "lastname", isActive = true });
List<Employee> EmployeesCopy = Employees.Select(x => new Employee(x)).ToList();
public class Employee
{
public string FirstName { get; set; }
public string LastName { get; set; }
public bool isActive { get; set; }
public Employee()
{ }
public Employee(Employee e)
{
FirstName = e.FirstName;
LastName = e.LastName;
isActive = e.isActive;
}
}
你在做一个浅拷贝,而不是深拷贝。这意味着新列表包含与原列表相同的对象。
要进行深度复制,您需要遍历原始列表并为新列表创建新的Employee
对象,如下所示。
private List<Employee> CloneEmployees(List<Employee> original)
{
var newList = new List<Employee>();
foreach (var employee in original)
{
newList.Add(new Employee
{
FirstName = employee.FirstName,
LastName = employee.LastName,
isActive = employee.isActive
});
}
return newList;
}
为什么对EmployeesCopy的isActive属性的更改也会修改原始列表?|
因为两个列表都指向Employee对象的同一个实例。您还需要深度复制Employee对象。
副本是一个新的List
对象,但它包含对原始列表中相同的Employee
对象集的引用。如果您希望两个列表中的Employee
对象是独立的,那么您必须单独复制它们并将副本放入新列表中。
您创建的副本只是列表的副本。不是对象的副本。也就是说,Employees[0] == EmployeesCopy[0]
.
因为使用new List<Employee>(Employees);
将给您List的新实例,而不是列表中包含的对象。您还应该考虑克隆列表中包含的对象,使用二进制序列化来序列化对象图。