可以从多个线程等待相同的任务 - 等待线程安全吗?

时间:2013-05-29 17:32:02

标签: c# asynchronous thread-safety async-await

是否等待线程安全?似乎Task类是线程安全的,所以我猜等待它也是线程安全的,但我还没有在任何地方找到确认。线程安全也是定制awaiter的要求 - 我的意思是IsCompleted,GetAwaiter等方法?即如果这些方法不是线程安全的,那么等待是否是线程安全的?但是,我不希望很快就需要定制的等待。

用户场景的一个例子:假设我有一个后台任务,它以异步方式返回一个结果,然后从多个线程中使用它:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;

namespace scratch1
{
    class Foo
    {
        Task<int> _task;

        public Foo()
        {
            _task = Task.Run(async () => { await Task.Delay(5000); return 5; });
        }

        // Called from multiple threads
        public async Task<int> Bar(int i)
        {
            return (await _task) + i;
        }
    }

    class Program
    {
        public static void Main(string[] args)
        {
            Foo foo = new Foo();

            List<Task> tasks = new List<Task>();
            foreach (var i in Enumerable.Range(0, 100))
            {
                tasks.Add(Task.Run(async () => { Console.WriteLine(await foo.Bar(i)); })); 
            }

            Task.WaitAll(tasks.ToArray());
        }
    }
}

1 个答案:

答案 0 :(得分:14)

正如您所提到的,await足够薄,以至于它首先看不到线程。

await(状态机中编译器生成的代码以及BCL中的支持类)相关联的代码一次只能在一个线程上运行。 (它可以在从等待状态返回时切换到不同的线程,但之前的运行已经完成)

实际的线程安全取决于您正在等待的对象 它必须有一种线程安全的方式来添加回调(或await一次两次可能会中断。)

此外,它必须只调用一次回调,否则可能会破坏状态机。 (特别是,如果它同时在两个线程上运行它的回调,事情会发生可怕的错误)

Task是线程安全的。