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

在C#中foreach循环内直接修改集合会抛出InvalidOperationException异常,应改用逆序for循环、缓存待操作元素、LINQ重建集合或Concurrent集合等安全方式。

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

使用 for 循环逆序遍历

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

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

示例:

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

缓存要操作的元素

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

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

示例:

List items = new List { "a", "b", "c", "d" };
List toRemove = new List();

foreach (string item in items) { if (item == "c") { toRemove.Add(item); } }

// 批量移除 foreach (string item in toRemove) { items.Remove(item); }

使用 LINQ 过滤重建集合

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

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

示例:

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

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

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

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

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

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