在for循环中启动任务:意外行为

时间:2016-08-05 06:05:59

标签: c# multithreading for-loop c#-4.0 task

我正在学习C#中的多线程。这让我很难,所以我正在做一些简单的程序来更好地理解它。

我意识到如果我有:

   static void Main()
    {
        Task[] tasks = new Task[4];
        for (int i = 0; i < 4; i++)
        {
            tasks[i] = Task.Factory.StartNew(() => {
                Console.WriteLine(i);
            });
        }
        Console.ReadKey();
    }

输出为4444,没有像我预期的那样0123。 为什么?

修改

Sadiq在他的回答中说,这种行为的原因是因为我关闭了一个循环变量。但是如果我在lambda语句之外和循环内部添加我的代码Thread.sleep(500);,我会得到0123。 那么,为什么这个由clousures引起的行为不会因添加这行代码而发生?

在我看来,被问及行为的原因是另一回​​事。如果你不理解我刚刚写的内容,那就是代码:

static void Main()
    {
        Task[] tasks = new Task[4];
        for (int i = 0; i < 4; i++)
        {
            tasks[i] = Task.Factory.StartNew(() => {
                Console.WriteLine(i);
            });
            Thread.Sleep(500); 
        }
        //Now the output is: 0123
        Console.ReadKey();
    }

1 个答案:

答案 0 :(得分:2)

创建一个局部变量 - 将值复制到其中,您应该得到预期的输出:

static void Main()
{
    Task[] tasks = new Task[4];
    for (int i = 0; i < 4; i++)
    {
        int x = i;
        tasks[x] = Task.Factory.StartNew(() => {
            Console.WriteLine(x);
        });
    }
    Console.ReadKey();
}

您获得意外输出的原因是因为i被共享。 You are closing over a loop variable