在循环中使用Task.Factory.StartNew

本文关键字:Task Factory StartNew 循环 | 更新日期: 2023-09-27 18:24:00

我正在努力掌握.NET.中的多线程编程

我想在for循环中创建100个任务,每个任务将休眠1秒,然后输出创建时使用的I及其托管线程id。在所有任务完成后,我想输出所用的总时间。

所以我运行这个简单的代码

    private static void Main(string[] args)
    {
        List<Task> tasks = new List<Task>();
        for (int i = 0; i < 100; i++)
        {
            tasks.Add(Task.Factory.StartNew(() =>
            {
                Thread.Sleep(1000);
                Console.WriteLine(i.ToString() + "  " + Thread.CurrentThread.ManagedThreadId);
            }));
        }
        Stopwatch watch = new Stopwatch();
        watch.Start();
        var final = Task.Factory.ContinueWhenAll(tasks.ToArray(), doneTasks =>
        {
            watch.Stop();
            Console.WriteLine("done in {0}", watch.Elapsed);
        });
        Console.ReadKey();
    }

我得到的输出:

100  24
100  23
...
100  14
100  25
done in 00:00:12.0044853

我不明白为什么所有的I都是一样的,为什么100?直觉上,我希望它们是0~99,但不是所有的100。

在循环中使用Task.Factory.StartNew

想象一下执行的顺序。您指示每个任务在一秒钟后输出i,但此时循环已完成执行,并且i等于100。

您可以使用以下StartNew重载来传递将由您的操作使用的state对象:

public Task StartNew(Action<object> action, object state)

并执行以下操作:

for (int i = 0; i < 100; i++)
{
    tasks.Add(Task.Factory.StartNew(o =>
    {
        int value = (int)o;
        Thread.Sleep(1000);
        Console.WriteLine(value.ToString() + "  " + Thread.CurrentThread.ManagedThreadId);
    }, i));
}

这是因为您的任务是在执行闭包之前创建的。如果在睡眠调用之前将I的值复制到一个变量中,或者更好的是,将其作为参数传递给您的操作,您应该会得到所需的结果。