在之前的博客2019-1-24-Task真的取消了么 - huangtengxiao介绍了task.run需要在方法执行过程中主动调用ThrowIfCancellationRequested()才能取消,这次我们研究下其他情况的取消场景


Task .Delay

我们知道Task.Dealy也有一个重载可以传入CancellationToken,我们做如下实验

        static void Main(string[] args)
        {
            var source = new CancellationTokenSource();
            Foo(source.Token);
            Thread.Sleep(TimeSpan.FromSeconds(3));
            source.Cancel();
            Console.WriteLine("取消任务");
            Console.ReadLine();
        }

        public static void Foo(CancellationToken token)
        {
            Task.Delay(TimeSpan.FromSeconds(5), token)
                .ContinueWith(task => Console.WriteLine(task.Status));
        }

和之前实验类似,只是Task.Run换成了Task.Delay。结果如图所示

1548584708789

可见Task.Delay可以直接响应取消

Parallel

我们再对Parallel做实验

        static void Main(string[] args)
        {
            var source = new CancellationTokenSource();
            Foo(source.Token);
            Thread.Sleep(TimeSpan.FromSeconds(3));
            source.Cancel();
            Console.WriteLine("取消任务");
            Console.ReadLine();
        }

        public static void Foo(CancellationToken token)
        {
            var options = new ParallelOptions()
            {
                CancellationToken = token,
            };
            Task.Run(() =>
            {
                Parallel.For(0, 10, options, (i, state) =>
                {
                    Console.WriteLine($"{i}任务开始");
                    Thread.Sleep(TimeSpan.FromSeconds(5));
                    Console.WriteLine($"{i}任务结束");
                });
            });
        }

通过Parallel.For创建10个并行任务,每个任务需要5秒钟,而第3秒时取消任务。结果如图所示

1548585794347

可见Parallel.For的行为和Task.Run是一致的

参考链接:


本文会经常更新,请阅读原文: https://xinyuehtx.github.io/post/Task%E7%9C%9F%E7%9A%84%E5%8F%96%E6%B6%88%E4%BA%86%E4%B9%88-2.html ,以避免陈旧错误知识的误导,同时有更好的阅读体验。

知识共享许可协议 本作品采用 知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议 进行许可。欢迎转载、使用、重新发布,但务必保留文章署名黄腾霄(包含链接: https://xinyuehtx.github.io ),不得用于商业目的,基于本文修改后的作品务必以相同的许可发布。如有任何疑问,请 与我联系