8

我开始学习 C# 5 和 .Net 4.5 中的异步编程,但有些东西我不明白。

private static int count;

public static void Main()
{
    LoopTest();

    Console.ReadKey(false);
}

static async void LoopTest()
{
    count = 0;

    for (var index = 0; index < 10; ++index)
    {
        Console.WriteLine( "({0}) In Loop before await, Index {1}, Thread: {2}", count++, index, Thread.CurrentThread.ManagedThreadId);
        await Task.Factory.StartNew(() => Thread.Sleep(10));
    }
}

结果是:

 (0) In Loop before await, Index 0, Thread: 9  
 (1) In Loop before await, Index 1, Thread: 10    
 (2) In Loop before await, Index 2, Thread: 11   
 (3) In Loop before await, Index 3, Thread: 10  
 (4) In Loop before await, Index 4, Thread: 11  
 (5) In Loop before await, Index 5, Thread: 10  
 (6) In Loop before await, Index 6, Thread: 12  
 (7) In Loop before await, Index 7, Thread: 11  
 (8) In Loop before await, Index 8, Thread: 10  
 (9) In Loop before await, Index 9, Thread: 12

那么是否有不同的线程访问同一个循环?索引变量是否存在竞争条件?

4

2 回答 2

13

不,没有竞争条件。您启动了一项新任务,该任务将在后台运行,但您的代码在该任务完成之前不会继续执行,因此您可以确保您的代码在任何时候都只会在一个线程中执行。

但是,因为这是一个控制台应用程序,所以没有当前的SyncronizationContext. (与 Winform、WPF、ASP、Silverlight 等应用程序相反。)这意味着任何await调用的延续都将在线程池中运行,而不是在调用线程中。所以是的,您在循环的每次迭代中都在一个线程池线程中运行,并且每次都可能是一个不同的线程,但是您可以确定,除了您在新任务中放入的任何内容之外,您将“在开始处理下一个线程之前,在任何给定线程上运行“完成”,这确保在任何给定时间只有一个线程运行代码(这意味着没有竞争条件)。

于 2013-01-17T15:01:21.967 回答
4

你在做什么并不是真正的异步

await Task.Factory.StartNew(() => Thread.Sleep(10));

这将创建一个任务,然后等待该任务完成。您也可以在这里调用 Thread.Sleep(10)。

更合适的方法是:

Task myTask = Task.Factory.StartNew(() => Thread.Sleep(10));
... do stuff here ...
await myTask; // Now wait for the task to complete

如果你想在一个循环中做一些任务,你应该看看Parallel.Foreach()方法。

于 2013-01-17T14:54:50.990 回答