是否可以在不知道当前正在运行的内容的情况下取消所有异步方法?
例如,我有几个可以运行异步任务的类:
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
}
}
是否可以执行此操作,而无需将任务保存到查询中?
答案 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
。