赶时间,请直接看最后
在开发过程中时常会遇到需要遍历集合并移除指定项的场景,我们用不同方法来尝试获得我们希望的结果。
首先声明将要用到的 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 会出现下标越界)。
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 方法会返回被移除数量