取消所有异步任务

Cancel all async tasks

是否可以取消所有异步方法,而不知道当前是什么运行ning?

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

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);
        }
    }
}

我想在注销之前关闭所有异步任务:

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
    }
}

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

你应该看看 CancellationTokenSource

您应该让您的 c1.SomeTask()c2.ContinuouslyTask() 接受取消令牌,并检查该令牌以查看请求是否正在被取消,如果是,则突然结束。

然后在您的 Program.Main() 中,它应该创建一个 CancellationTokenSource 并将 CancellationTokenSource.Token 传递给它调用的 2 个异步方法。 Program.Logout() 应该可以访问此 CancellationTokenSource,以便它可以在注销时发出取消。参见示例 here

注意:通常建议异步方法 return 任务,而不是 void。此外,异步方法命名为 xxxAsync(例如 DoWorkAsync)是一种约定。

这基本上是在扩展@FrankFajardo 的回答以提供具体示例。 当你传入一个 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);
        }
    }
}

现在当你想取消它时,你只需在你的代码中添加 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();
}

注意我使用 Task.WaitAll 只是因为这是一个控制台应用程序,其中 Main 不能异步。否则,使用 Task.WhenAll which returns 等待 Task.