List<T>.ForEach(Action<T> action) 是 C# 中 List<T> 类提供的一个实例方法,用于对列表中的每个元素执行指定的操作。
1、基本用法
该方法接受一个 Action<T> 委托作为参数,通常使用 Lambda 表达式编写逻辑。
List<string> names = new List<string> { "Alice", "Bob", "Charlie" };
// 使用 Lambda 表达式
names.ForEach(item => Console.WriteLine(item));
// 使用匿名方法(旧式写法)
names.ForEach(delegate(string name) {
Console.WriteLine(name);
});
2、特性与限制
不支持中断控制:在 ForEach 的 Lambda 表达式中,不能使用 break、continue 或 return(指跳出整个循环,而非仅退出当前委托)。如果需要提前终止遍历,必须使用传统的 foreach 语句或 for 循环 。
线程安全与集合修改:虽然 List<T>.ForEach 内部实现允许在遍历时修改集合(如添加或删除元素),但这会导致索引偏移,可能引发逻辑错误、重复处理或死循环。官方文档明确警告:不支持在委托正文中修改基础集合,这会导致未定义的行为 。
异步支持有限:不要在 ForEach 中使用 async void 或等待异步操作(如 list.ForEach(async item => await DoSomething(item))),这会掩盖异常并导致调试困难。如需并行或异步处理,应使用 Parallel.ForEach 或手动编写 foreach 配合 Task 。
3、性能对比
轻微开销:List<T>.ForEach 本质上是封装了 for 循环加上委托调用。每次迭代都需要通过 Action<T> 间接执行,存在额外的跳转和可能的分配开销。在绝大多数业务场景下,这种差异可以忽略不计;但在极高频率的微基准测试或百万级数据遍历中,传统 for 循环或 foreach 关键字通常略快 。
JIT 优化限制:由于涉及委托调用,JIT 编译器通常无法将其内联(inline),而简单的 for 循环更容易被优化 。
4、建议
总结:List<T>.ForEach 适合快速编写简单的同步遍历逻辑,但在需要流程控制、高性能或异步操作的场景中,传统的 foreach 或 for 循环是更安全、更灵活的选择。