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

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

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

在 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 中修改的问题,仍需谨慎使用迭代器。

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