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

在 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 中修改的问题,仍需谨慎使用迭代器。
基本上就这些。选择哪种方式取决于你的具体需求:是否需要保留原集合引用、是否涉及多线程、性能要求如何。关键是不要在枚举器运行时直接改动原集合结构。