C# 中的 foreach
循环和 Lambda 表达式线程安全问题
让我们分析以下代码片段:
<code class="language-csharp">foreach (Foo f in ListOfFoo) { Thread thread = new Thread(() => f.DoSomething()); threads.Add(thread); thread.Start(); }</code>
<code class="language-csharp">foreach (Foo f in ListOfFoo) { Foo f2 = f; Thread thread = new Thread(() => f2.DoSomething()); threads.Add(thread); thread.Start(); }</code>
哪个代码片段能够确保每个线程都调用与线程创建时相同的循环迭代中的 Foo
实例上的方法?
在 C# 5 及更高版本中,由于编译器对闭包作用域内变量定义的更改,这两个代码片段都是安全的。但是,在 C# 5 之前,只有第二个代码片段是安全的。
在第一个代码片段中,变量 f
在循环外部声明,并在整个执行过程中保持可见。这意味着闭包作用域中只有一个 f
实例。因此,访问 f
的不同线程可能会发生冲突,导致方法调用错误。
为了缓解这个问题,第二个代码片段在循环内声明了一个新变量 f2
。这确保了每个闭包作用域都有其对 Foo
实例的引用副本,从而保证了每个线程的安全方法执行。
下面的示例演示了这个问题:
<code class="language-csharp">static void Main() { int[] data = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9 }; foreach (int i in data) { new Thread(() => Console.WriteLine(i)).Start(); } Console.ReadLine(); }</code>
如果不使用临时变量,这段代码将产生不可预测和不正确的输出。 这是因为所有线程都引用同一个 i
变量,而 foreach
循环结束后,i
的值是最后一个值。
因此,为了确保线程安全,在 foreach
循环中使用 Lambda 表达式时,最好始终创建局部变量的副本,如同第二个代码片段所示。
以上是我的'foreach”循环和 Lambda 线程在 C# 中安全吗?的详细内容。更多信息请关注PHP中文网其他相关文章!