为什么列表的副本仍然使用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;

为什么改变EmployeesCopyisActive属性也会改变原列表中的属性?

为什么列表的副本仍然使用c#改变原始列表的属性?

因为新列表仍然包含对相同雇员对象的引用。您可以在新列表中创建新列表,如下所示:

    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的新实例,而不是列表中包含的对象。您还应该考虑克隆列表中包含的对象,使用二进制序列化来序列化对象图。