C# 如何在循环中安全地修改集合 - 避免“集合已修改”异常

来源:这里教程网 时间:2026-02-21 17:35:06 作者:

在 C# 中,直接在 foreach 循环中修改集合(如添加或删除元素)会触发“集合已修改;枚举操作可能无法执行”异常。这是因为 .NET 的集合在遍历时会检测内部版本号,一旦发现被修改就抛出 InvalidOperationException。要安全地修改集合,需要采用合适的方法。

使用 for 循环逆序遍历

当你需要删除元素时,推荐使用 for 循环并从集合末尾向前遍历。这样即使删除元素也不会影响未访问的索引。

适用于 List、数组等支持索引访问的集合 避免因索引偏移导致跳过元素或越界

示例:

List<string> items = new List<string> { "a", "b", "c", "d" };
for (int i = items.Count - 1; i >= 0; i--)
{
    if (items[i] == "b")
    {
        items.RemoveAt(i);
    }
}

缓存要操作的元素

先用 foreach 遍历集合,将需要删除或处理的元素暂存到另一个集合中,再进行批量修改。

逻辑清晰,适合复杂判断条件 避免在迭代器活跃时修改原集合

示例:

List<string> items = new List<string> { "a", "b", "c", "d" };
List<string> toRemove = new List<string>();
<p>foreach (string item in items)
{
if (item == "c")
{
toRemove.Add(item);
}
}</p><p>// 批量移除
foreach (string item in toRemove)
{
items.Remove(item);
}

使用 LINQ 过滤重建集合

如果允许创建新集合,可以用 Where 等方法生成过滤后的结果,替代原集合。

代码简洁,函数式风格 适合不需要修改原引用的场景

示例:

List<string> items = new List<string> { "a", "b", "c", "d" };
items = items.Where(x => x != "b").ToList();

使用支持并发修改的集合类型

在多线程或频繁修改场景下,可考虑使用 System.Collections.Concurrent 命名空间中的集合,如 ConcurrentBag、ConcurrentQueue 等。

线程安全,允许多个写入/读取操作 不支持 foreach 中删除,但提供 TryAdd/TryTake 等安全方法

注意:Concurrent 集合不能完全解决 foreach 中修改的问题,仍需谨慎使用迭代器。

基本上就这些。选择哪种方式取决于你的具体需求:是否需要保留原集合引用、是否涉及多线程、性能要求如何。关键是不要在枚举器运行时直接改动原集合结构。

相关推荐