Task.Factory.StartNew和Task.Run到底有什么区别?

前言

Task.Factory.StartNew 和 Task.Run 都可以创建 Task:

Task.Factory.StartNew(() => { Console.WriteLine("Task.Factory.StartNew"); });

Task.Run(() => { Console.WriteLine("Task.Run"); });            

那它们之间有什么区别呢?.

实现代码

查看这 2 个方法的内部实现,其内部实现逻辑其实是一样的,只是传的默认参数不同:

//Task.Factory.StartNew
public Task StartNew(Action action)
{
    Task? currTask = Task.InternalCurrent;
    return Task.InternalStartNew(currTask, action, null, m_defaultCancellationToken, GetDefaultScheduler(currTask),
        m_defaultCreationOptions, InternalTaskOptions.None);
}
//Task.Runpublic static Task Run(Action action)
{
    return Task.InternalStartNew(null, action, null, default, TaskScheduler.Default,
        TaskCreationOptions.DenyChildAttach, InternalTaskOptions.None);
}

最关键的参数区别是 Task.Run 传入了 TaskCreationOptions.DenyChildAttach

那这个参数有什么用呢?

DenyChildAttach

查看官方文档的解释,DenyChildAttach 的作用是阻止子任务附加到其父任务

设想下从 Task 对象调用第三方库组件的应用。如果第三方库组件也创建一个 Task 对象,并指定 TaskCreationOptions.AttachedToParent 以将其附加到父任务中,则子任务中出现的任何未经处理的异常将会传播到父任务。这可能会导致主应用中出现意外行为。

创建代码验证一下:

Stopwatch stopwatch1 = new Stopwatch();
stopwatch1.Start();
var task1 = Task.Factory.StartNew(() =>
{
    Run();

    Console.WriteLine("Task.Factory.StartNew");
});

await task1;
stopwatch1.Stop();
Console.WriteLine(stopwatch1.ElapsedMilliseconds);

Stopwatch stopwatch2 = new Stopwatch();
stopwatch2.Start();
var task2 = Task.Run(() =>
{
    Run();
    Console.WriteLine("Task.Run");
});

await task2;
stopwatch2.Stop();
Console.WriteLine(stopwatch2.ElapsedMilliseconds);

Run 方法代表执行相同的第三方库组件调用,内部使用了 AttachedToParent

private static void Run()
{
    Task.Factory.StartNew(() =>
    {
        Thread.Sleep(1000);
        Console.WriteLine("Run");
    }, TaskCreationOptions.AttachedToParent);
}

运行程序,你将会看到类似的如下输出:

Task.Factory.StartNew
Run
1080
Task.Run
1
Run

使用 Task.Factory.StartNew 必须等待 AttachedToParent 任务执行完,而 Task.Run 不必。

结论

一般情况下,尽量使用 Task.Run,如果需要更精细地控制任务的行为,比如 TaskCreationOptions, 才使用 Task.Factory.StartNew