5

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

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

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

是否可以做到这一点,而无需将任务保存到查询中?

4

2 回答 2

3

您应该查看CancellationTokenSource

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

然后在你的Program.Main(),它应该创建一个 CancellationTokenSource 并将 CancellationTokenSource.Token 传递给它调用的 2 async 方法。此 CancellationTokenSource 应该可供 CancellationTokenSource 访问,Program.Logout()以便它可以在注销时发出取消。在此处查看示例

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

于 2016-01-16T13:15:17.583 回答
3

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

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.WhenAllwhich 返回一个 awaitable Task

于 2016-01-16T14:25:09.960 回答