详解 .NET Core 遍历 List 并移除项

赶时间,请直接看最后

在开发过程中时常会遇到需要遍历集合并移除指定项的场景,我们用不同方法来尝试获得我们希望的结果。

首先声明将要用到的 list 对象。

var list = new List<string>() { "Item1", "Item2", "Item3", "Item3-1", "Item3-2" };
  • 方法1:很大可能性会这些写,下面的代码看起来完全没问题,编译也能顺利通过。只有在运行的时候才会发现问题:
foreach (var item in list)
{
    if (item.Contains("3"))
        list.Remove(item);
}

报错信息:System.InvalidOperationException:“Collection was modified; enumeration operation may not execute.”

  • 方法2:改成 for 以后能正常运行了。结果会发现“Item3-1,Item3-3”也输出了,我们希望是将包含“3”的项移除,结果看起来没有。
for (int i = 0; i < list.Count; i++)
{
    var item = list[i];
    if (item.Contains("3"))
        list.Remove(item);
}

Console.WriteLine(string.Join(",", list));

输出:Item1,Item2,Item3-1,Item3-3

循环逻辑重现,可用从下表看出循环只执行了4轮,原因是 list 被移除了两条后,长度缩短导致位置出现改变:

下标/值 是否满足条件 length
1 0 / Item1 false 6
2 1 / Item2 false 6
3 2 / Item3 true 5
4 3 / Item3-2 true 4
  • 方法3:将 length 提出来作为变量,这样就能将整个 list 遍历完了。但是这样会导致下标越界。
var length = list.Count;
for (int i = 0; i < length; i++)
{
    var item = list[i];
    if (item.Contains("3"))
        list.Remove(item);
}

报错信息:System.ArgumentOutOfRangeException:“Index was out of range. Must be non-negative and less than the size of the collection.

  • 方法4:将 list 复制给新变量,然后再使用遍历变量新,操作 list。
var listNew = list;
foreach (var item in listNew)
{
    if (item.Contains("3"))
        list.Remove(item);
}

报错信息:System.InvalidOperationException:“Collection was modified; enumeration operation may not execute.”

和方法一一样的错误,因为 list 是一个引用类型,listNew 与 list 共用同一个内存地址,所以会报错(改成 for 会出现下标越界)。

  • 方法5:将 list clone 一次,遍历 clone 对象时删除 list 中的子项。
var clone = list.Clone();
foreach (var item in clone)
{
    if (item.Contains("3"))
        list.Remove(item);
}

Console.WriteLine(string.Join(",", list));

输出:Item1,Item2

这是我们希望得到的结果。

通过 clone 的方法将对象从一个内存地址变为两个,通过下面可以看出两个对象的关系,在 clone 后Ta们就不相等了。

var clone = list;
Console.WriteLine(clone == list);                       //true
Console.WriteLine(ReferenceEquals(clone, list));        //true
Console.WriteLine(ReferenceEquals(clone, list.Clone()));//false
  • 方法6:使用 LINQ RemoveAll 方法可以更简单
var count = list.RemoveAll(x => x.Contains("3"));
Console.WriteLine(count);
Console.WriteLine(string.Join(",", list));

输出:
4
Item1,Item2

RemoveAll 方法会返回被移除数量

发布了87 篇原创文章 · 获赞 69 · 访问量 60万+

猜你喜欢

转载自blog.csdn.net/S2T11Enterprise/article/details/101909801