取消所有异步任务

Ser*_*bin 5 c# asynchronous async-await cancellation-token

是否可以在不知道当前正在运行的情况下取消所有异步方法?

例如,我有几个可能运行异步任务的类:

class Class1
{
    public async void SomeTask()
    {
        for (int i = 0; i < 5; i++)
        {
            // Doing some job
            await Task.Delay(2000);
        }
    }
}

class Class2
{
    public async void ContinuouslyTask()
    {
        for (;;)
        {
            // Doing some job on background
            await Task.Delay(1000);
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

我想在注销之前关闭每个异步任务:

class Program
{
    static void Main(string[] args)
    {
        var p = new Program();
        var c1 = new Class1();
        var c2 = new Class2();

        c1.SomeTask();
        c2.ContinuouslyTask(); 

        while (Console.ReadKey().Key != ConsoleKey.Enter) { }

        p.Logout();
    }

    private void Logout()
    {
        // Cancel all async tasks

        // And do logout work
    }
}
Run Code Online (Sandbox Code Playgroud)

是否可以在不将任务保存到查询中的情况下执行此操作?

Yuv*_*kov 4

这基本上是扩展@FrankFajardo 的答案以提供一个具体的例子。当您传入 a 时CancellationToken,您还需要监视它是否有来自外部的取消请求。它看起来像这样:

class Class1
{
    public async Task SomeTaskAsync(CancellationToken cancellationToken)
    {
        for (int i = 0; i < 5; i++)
        {
            if (cancellationToken.IsCancellationRequested)
                break;
            // Doing some job
            await Task.Delay(2000);
        }
    }
}

class Class2
{
    public async Task ContinuouslyTaskAsync(CancellationToken cancellationToken)
    {
        while (!cancellationToken.IsCancellationRequested)
        {
            // Doing some job on background
            await Task.Delay(1000);
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

现在,当您想取消它时,只需CancellationTokenSource.Cancel()在代码中添加调用即可:

static void Main(string[] args)
{
    var p = new Program();
    var c1 = new Class1();
    var c2 = new Class2();

    var cancellationTokenSource = new CancellationTokenSource();
    var someTask = c1.SomeTask(cancellationTokenSource.Token);
    var continuousTask = c2.ContinuouslyTask(cancellationTokenSource.Token); 

    while (Console.ReadKey().Key != ConsoleKey.Enter) { }

    cancellationTokenSource.Cancel();
    Task.WaitAll(someTask, continuousTask);

    p.Logout();
}
Run Code Online (Sandbox Code Playgroud)

注意我使用它只是因为这是一个不能异步的Task.WaitAll控制台应用程序。Main否则,使用Task.WhenAll它返回一个可等待的Task