我尝试在取消下面的任务时运行一个简单的例子
- CancellationTokenSource tokenSource2 = new CancellationTokenSource();
- CancellationToken token2 = tokenSource2.Token;
- Task task2 = new Task(() =>
- {
- for (int i = 0; i < int.MaxValue; i++)
- {
- token2.ThrowIfCancellationRequested();
- Thread.Sleep(100);
- Console.WriteLine("Task 2 - Int value {0}",i);
- }
- },token2);
- task2.Start();
- Console.WriteLine("Press any key to cancel the task");
- Console.ReadLine();
- tokenSource2.Cancel();
- Console.WriteLine("Task 2 cancelled? {0}",task2.IsCanceled);
我期望Console.WriteLine(“任务2取消?{0}”,task2.IsCanceled);将打印**“任务2取消?真”**,但它打印“假”.
你知道发生了什么吗?这是预期的行为吗?谢谢.
解决方法
首先,也许你误解了IsCanceled的意思?这并不意味着“此任务正在等待取消,因此它应该很快完成”,这意味着“此任务已被取消,现在已完成”.
如果你没有误解,请考虑事件的顺序是什么.这是怎么回事:
>调用ThrowIfCancellationRequested(),但令牌尚未取消,因此它不会抛出.
>调用Thread.Sleep(),因此运行Task的线程会休眠.
>调用Cancel().
>检查IsCanceled.任务中的代码没有机会意识到令牌已被取消,因此它仍在运行,因此IsCanceled返回false.
>再次调用ThrowIfCancellationRequested(),这次抛出,实际上取消了Task.
这就是为什么ISCanceled会向您返回false.如果你想让它返回true,你可以在检查IsCanceled之前添加类似Thread.Sleep(150)的东西,或者更好的是,实际上等待Task完成.