目录
前言
实现代码
DenyChildAttach
结论
前言
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查看官方文档[1]的解释,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
。