有人可以解释为什么 SecondTask 和加载事件在给定这段代码的情况下不更新 UI 吗?
public Task First()
{
return Task.Run(
() =>
{
Task.Delay(4000);
Dispatcher.Invoke(() =>
{
Title = "From First()";
});
});
}
public Task Second()
{
return Task.Run(
() =>
{
Task.Delay(5000);
Dispatcher.Invoke(() =>
{
Title = "From Second()";
});
Task.WaitAny(First());
});
}
private void ParentWindow_Loaded(object sender, RoutedEventArgs e)
{
Second();
Title = "From Event";
}
当我运行此代码时,由于调用 First(),只有表单的标题仅更改一次。
没有执行任何其他操作(至少从我所见,我可以看到已到达断点)。
我也尝试过使用 Task.WhenAny(First());
而不是 Task.WaitAny(First());
最佳答案
问题是 Task.Delay
没有被等待。 Task.Delay
是一个异步休眠,所以它会直接返回,如果你想在延迟后真正继续执行,你需要 await
它。
Second()
方法中发生的是 lambda 表达式在后台线程上执行。当 Task.Delay(5000)
被调用时,调用立即返回,因为它没有被等待,Title
然后被设置为 From Second()
.
public Task Second()
{
return Task.Run(
() =>
{
Task.Delay(5000); // need to await here
Dispatcher.Invoke(() =>
{
Title = "From Second()";
});
Task.WaitAny(First());
});
}
First()
然后被调用,因为 Task.Delay(4000)
也没有等待,它立即将 Title
设置为来自 First()
。所以最终的结果是你只会看到 From First()
。
public Task First()
{
return Task.Run(
() =>
{
Task.Delay(4000); // need to await here
Dispatcher.Invoke(() =>
{
Title = "From First()";
});
});
}
您可能还想在事件处理程序中等待
调用Second()
,除非您打算先将标题设置为From Event
然后用 Second()
的结果覆盖它。
private void ParentWindow_Loaded(object sender, RoutedEventArgs e)
{
Second(); // await this?
Title = "From Event";
}
关于.net - Task.Run 代码不执行,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/23600440/