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.我错过了什么?

谢谢

C# “等待”实际上返回什么

我建议您阅读我的async简介。此外,LinqPad 内置了一个出色的教程(示例 ->下载/导入更多示例 -> C# 5 交互式教程中的异步)。

async代码中的任务标识符可能有点棘手。发生的情况是,每个async Task方法都会创建自己的任务来表示该方法调用。因此,Task.Run创建一个Task并将其返回到Step2Step1(测试中的 id 17)。 Step1还会创建一个Task并将其返回到Step0(测试中的 id 18)。 Step0 还创建一个Task并将其返回给StartItStartIt 还创建一个Task并将其返回给Main

因此,Task.Run返回的Task(与Step2返回的Task相同)将被取消。 Step1捕获已取消的异常,因此它正常返回,并且Step1返回的Task不会被取消。

如果要传播异常,请不要捕获已取消的异常(或者,如果确实要捕获它进行记录,则通过throw;重新抛出它)。