Task Parallel Library 很棒,在过去的几个月里我经常使用它。但是,有一些事情真的让我很困扰:这TaskScheduler.Current
是默认任务调度程序,而不是TaskScheduler.Default
. 乍一看,这在文档或示例中绝对不明显。
Current
可能会导致细微的错误,因为它的行为会根据您是否在另一个任务中而改变。哪个不容易确定。
假设我正在编写一个异步方法库,使用基于事件的标准异步模式在原始同步上下文上发出信号完成信号,就像 XxxAsync 方法在 .NET Framework 中所做的一样(例如DownloadFileAsync
)。我决定使用任务并行库来实现,因为用下面的代码很容易实现这个行为:
public class MyLibrary
{
public event EventHandler SomeOperationCompleted;
private void OnSomeOperationCompleted()
{
SomeOperationCompleted?.Invoke(this, EventArgs.Empty);
}
public void DoSomeOperationAsync()
{
Task.Factory.StartNew(() =>
{
Thread.Sleep(1000); // simulate a long operation
}, CancellationToken.None, TaskCreationOptions.None, TaskScheduler.Default)
.ContinueWith(t =>
{
OnSomeOperationCompleted(); // trigger the event
}, TaskScheduler.FromCurrentSynchronizationContext());
}
}
到目前为止,一切正常。现在,让我们在 WPF 或 WinForms 应用程序中单击按钮时调用此库:
private void Button_OnClick(object sender, EventArgs args)
{
var myLibrary = new MyLibrary();
myLibrary.SomeOperationCompleted += (s, e) => DoSomethingElse();
myLibrary.DoSomeOperationAsync(); // call that triggers the event asynchronously
}
private void DoSomethingElse() // the event handler
{
//...
Task.Factory.StartNew(() => Thread.Sleep(5000)); // simulate a long operation
//...
}
Task
在这里,编写库调用的人选择在操作完成时开始新的调用。没有什么不寻常的。他或她遵循网络上随处可见的示例,并且Task.Factory.StartNew
无需指定即可简单地使用TaskScheduler
(并且在第二个参数处指定它并不容易重载)。该DoSomethingElse
方法在单独调用时工作正常,但一旦被事件调用,UI 就会冻结,因为TaskFactory.Current
将重用我的库延续中的同步上下文任务调度程序。
找出这可能需要一些时间,特别是如果第二个任务调用被埋在一些复杂的调用堆栈中。当然,一旦你知道一切是如何工作的,这里的修复很简单:总是TaskScheduler.Default
为你期望在线程池上运行的任何操作指定。但是,也许第二个任务是由另一个外部库启动的,不知道这种行为并且StartNew
在没有特定调度程序的情况下天真地使用。我预计这种情况会很普遍。
在把我的脑袋绕过去之后,我无法理解编写 TPL 的团队选择使用TaskScheduler.Current
而不是TaskScheduler.Default
作为默认值:
- 一点都不明显,
Default
不是默认的!而且文档严重缺乏。 - 真正使用的任务调度器
Current
依赖于调用栈!这种行为很难保持不变量。 - 指定任务调度程序很麻烦,
StartNew
因为您必须首先指定任务创建选项和取消令牌,导致行长、可读性差。这可以通过编写扩展方法或创建TaskFactory
使用Default
. - 捕获调用堆栈会带来额外的性能成本。
- 当我真的希望一个任务依赖于另一个正在运行的父任务时,我更愿意明确指定它以简化代码阅读,而不是依赖调用堆栈魔术。
我知道这个问题听起来可能很主观,但我找不到一个好的客观论据来解释为什么这种行为会如此。我确定我在这里遗漏了一些东西:这就是我转向你的原因。