.NET CancellationTokenSource 取消任务用法详解
Cancel不代表终止代码
不要想当然地依靠Cancel机制“terminate a Task”,这是做不到的!
一个常见的误解是取消了一个任务后,该任务就会自动停止;但其实这种理解是不正确的;试想一下,如果取消真的可以达到terminate一个Task的效果,那它和危险的Thread.Abort有什么区别呢?因为真的按照“cancel即在Task代码内部抛出异常”去理解,那么你没有对结束后的程序状态有任何扫尾工作。
所以CancellationTokenSource抑或其对应的CancellationToken都只是发一个通知给Task:你可以结束了;而具体怎样捕获这个消息并响应,是Task的用户代码自己思考的问题。
参考原文中的代码,关键的一个片段是:
if (ct.IsCancellationRequested)
{
Console.WriteLine("Task {0} cancelled", taskNum);
ct.ThrowIfCancellationRequested();
}
这类似于软件工程中的“埋点”,即在关键的代码行中插入监控代码,判断任务是否被取消
WhenAny或Wait(token)会导致异常抛出
看回原文中的代码:
try
{
await Task.WhenAll(tasks.ToArray());
}
catch (OperationCanceledException)
{
Console.WriteLine($"\n{nameof(OperationCanceledException)} thrown\n");
}
一旦任务取消,那么Task.WhenAll会导致异常抛出;这里是非常容易误导人的一点:Task内部也发生了异常,从而打断的Task内部代码流,从而令任务强制退出了吗?
答案是:Task本身代码运行只受到自己内部代码控制,不可能单单因为CancellationTokenSource发出了取消信号,就终端了任务的运行。
作者:虾米哥
微信公众号:IT虾米,左侧为二维码
个人技术网站-IT虾米网:http://www.itxm.cn
个人技术网站-编程符号网:http://www.itfh.cn
个人技术网站-IT干货网:http://www.itgh.cn
新浪微博:https://weibo.com/u/2814576687
如果你想及时得到个人撰写文章以及著作的消息推送,或者想看看个人推荐的技术资料,可以扫描左边二维码(或者长按识别二维码)关注个人公众号。
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。

浙公网安备 33010602011771号