任务Task系列之使用CancellationToken取消Task
本文参考书籍《CLR via C#》
Task的取消采用一种形如令牌(Token)的方式。首先先构建一个CancellationTokenSource实例,然后任务中执行的方法必须能接受一个CancellationToken类型的参数。
假设我们有这样一个方法:
private static Int32 NumValue(CancellationToken token, Int32 n) { Int32 sum = 0; for (int i = 0; i < n; i++) { token.ThrowIfCancellationRequested(); checked { sum += i; } } return sum; }
上述代码中的循环中调用CancellationToken的ThrowIfCancellationRequested()方法定时检查操作是否已经取消,这个方法与CancellationToken的IsCancellationRequested属性作用一致,如果CancellationTokenSource已经取消,ThrowIfCancellationRequested()会抛出一个异常表示当前任务已经被取消。
我们创建Task和CancellationTokenSource的对象:
private static void TaskRun() { CancellationTokenSource tokenSource = new CancellationTokenSource(); Task<Int32> task = Task.Run(() => NumValue(tokenSource.Token, 1000000), tokenSource.Token); //发出取消的请求 tokenSource.Cancel(); try { Console.WriteLine("计算求和为:{0}",task.Result); } catch (AggregateException ex) { ex.Handle(e => e is OperationCanceledException); Console.WriteLine("用户已取消"); } }
可在创建Task时将一个CancellationToken传给构造器,从而将两者相关联,如果CancellationToken在Task调度前取消,那么Task就会被取消,永远都不执行。但如果Task已调度,那么Task的代码就只支持显示取消,其操作才能在执行期间取消,遗憾的是,虽然Task关联了一个CancellationToken,但却没有办法访问他。因此,必须在Task的代码中获得创建Task对象时的同一个CancellationToken。为此,最简单的办法就是使用一个Lamda表达式,将CancellationToken作为闭包变量传递。