C# “等待”实际上返回什么
本文关键字:返回 什么 实际上 等待 | 更新日期: 2023-09-27 18:31:07
下面是一些在 LinqPad 中"按原样"运行的代码。这表明我对await
正在做的事情的误解。进一步查看结果和我的问题。
void Main()
{
var cts = new CancellationTokenSource();
StartIt(cts.Token);
Thread.Sleep(500);
"Cancelling".Dump();
cts.Cancel();
}
// Define other methods and classes here
async Task StartIt(CancellationToken token)
{
"StartIt".Dump();
var o = new TestClass();
await o.Step0(token);
}
public class TestClass
{
public async Task Step0(CancellationToken token)
{
var t = Step1(token);
("Step0.Task.Id => " + t.Id).Dump();
await t;
("Step0.IsCancelled? " + t.IsCanceled).Dump();
}
public async Task Step1(CancellationToken token)
{
var t = Step2(token);
("Step1.Task.Id => " + t.Id).Dump();
try {
await t;
}
catch (OperationCanceledException)
{
"Step1.OperationCanceledException".Dump();
}
("Step1.IsCancelled? " + t.IsCanceled).Dump();
}
public Task Step2(CancellationToken token)
{
return Task.Run(() =>{
Thread.Sleep(3000);
token.ThrowIfCancellationRequested();
"Done".Dump();
}, token);
}
}
所以上面产生:
StartIt
Step1.Task.Id => 17
Step0.Task.Id => 18
Cancelling
Step1.OperationCanceledException
Step1.IsCancelled? True
Step0.IsCancelled? False
我的问题:
我期待 Step0 和 Step1 的任务。ID 相同。(因此)我希望发生的取消会传播这些方法,以便我可以询问t.IsCanceled
并采取适当的行动。
但是我得到了一个不同的Task
被送回Step0
.我错过了什么?
谢谢
我建议您阅读我的async
简介。此外,LinqPad 内置了一个出色的教程(示例 ->下载/导入更多示例 -> C# 5 交互式教程中的异步)。
async
代码中的任务标识符可能有点棘手。发生的情况是,每个async Task
方法都会创建自己的任务来表示该方法调用。因此,Task.Run
创建一个Task
并将其返回到Step2
和Step1
(测试中的 id 17
)。 Step1
还会创建一个Task
并将其返回到Step0
(测试中的 id 18
)。 Step0
还创建一个Task
并将其返回给StartIt
,StartIt
还创建一个Task
并将其返回给Main
。
因此,Task.Run
返回的Task
(与Step2
返回的Task
相同)将被取消。 Step1
捕获已取消的异常,因此它正常返回,并且Step1
返回的Task
不会被取消。
如果要传播异常,请不要捕获已取消的异常(或者,如果确实要捕获它进行记录,则通过throw;
重新抛出它)。