取消所有异步任务

时间:2016-01-16 12:21:16

标签: 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);
        }
    }
}

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

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

是否可以执行此操作,而无需将任务保存到查询中?

2 个答案:

答案 0 :(得分:3)

你应该看看CancellationTokenSource

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

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

注意:通常建议异步方法返回Tasks,而不是void。此外,异步方法被命名为xxxAsync(例如DoWorkAsync)。

答案 1 :(得分:3)

这基本上是扩展@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返回等待Task