介绍

    为什么需要CancellationToken?因为Task没有方法支持在外部取消Task,只能通过一个公共变量存放线程的取消状态,在线程内部通过变量判断线程是否被取消,当CancellationToken是取消状态,Task内部未启动的任务不会启动新线程。

  取消令牌(CancellationToken) ,正确并合理的使用 CancellationToken 可以让业务达到简化代码、提升服务性能的效果;当在业务开发中,需要对一些特定的应用场景进行深度干预的时候,CancellationToken 将发挥非常重要的作用。

1. 任务被取消时执行某个操作

var tokenSource = new CancellationTokenSource();
tokenSource.Token.Register(() => { Console.WriteLine("线程被取消"); });

2. 延时取消,对长时间阻塞调用的异步取消令牌应用

在某些场景中,我们需要请求外部的第三方资源,比如请求天气预报信息;但是,由于网络等原因,可能会造成长时间的等待以致业务超时退出,这种情况可以使用 CancellationToken 来进行优化,但请求超过指定时长后退出,而不必针对每个 HttpClient 进行单独的超时设置

        public async static Task GetToday()
{
CancellationTokenSource cts = new CancellationTokenSource();
cts.CancelAfter(3000);
HttpClient client = new HttpClient();
var res = await client.GetAsync("http://www.weather.com.cn/data/sk/101110101.html", cts.Token);
var result = await res.Content.ReadAsStringAsync();
Console.WriteLine(result); cts.Dispose();
client.Dispose();
}

3. CancellationToken 的链式反应

可以使用创建一组令牌,通过链接各个令牌,使其建立通知关联,当 CancellationToken 链中的某个令牌收到取消通知的时候,由链式中创建出来的 CancellationToken 令牌也将同时取消

3.1 创建链式测试代码
public async static Task Test()
{
CancellationTokenSource cts1 = new CancellationTokenSource();
CancellationTokenSource cts2 = new CancellationTokenSource();
var cts3 = CancellationTokenSource.CreateLinkedTokenSource(cts1.Token, cts2.Token); cts1.Token.Register(() =>
{
Console.WriteLine("cts1 Canceling");
});
cts2.Token.Register(() =>
{
Console.WriteLine("cts2 Canceling");
});
cts2.CancelAfter(1000); cts3.Token.Register(() =>
{
Console.WriteLine("root Canceling");
}); var res = await new HttpClient().GetAsync("http://www.weather.com.cn/data/sk/101110101.html", cts1.Token);
var result = await res.Content.ReadAsStringAsync();
Console.WriteLine("cts1:{0}", result); var res2 = await new HttpClient().GetAsync("http://www.weather.com.cn/data/sk/101110101.html", cts2.Token);
var result2 = await res2.Content.ReadAsStringAsync();
Console.WriteLine("cts2:{0}", result2); var res3 = await new HttpClient().GetAsync("http://www.weather.com.cn/data/sk/101110101.html", cts3.Token);
var result3 = await res2.Content.ReadAsStringAsync();
Console.WriteLine("cts3:{0}", result3);
}

上面的代码定义了 3 个 CancellationTokenSource ,分别是 cts1,cts2,cts3,每个 CancellationTokenSource 分别注册了 Register 取消回调委托,然后,使用 HttpClient 发起 3 组网络请求;其中,设置 cts2 在请求开始 1秒 后退出,预期结果为:当 cts2 退出后,由于 cts3 是使用 CreateLinkedTokenSource(cts1.Token, cts2.Token) 创建出来的,所以 cts3 应该也会被取消,实际上,无论 cts1/cts2 哪个令牌取消,cts3 都会被取消

最新文章

  1. dragsort html拖拽排序
  2. 在Spring-Mybatis-Restful中配置多数据源的properties文件
  3. yii2-搜索带分页,分页的两种方式
  4. servlet filter可以用注解
  5. Nagios:企业级系统监控方案
  6. 一个用WPF做的简单计算器源代码
  7. Linux 学习笔记 查看文件内容诸多命令
  8. JVM之---Java内存分配参数(第四篇)
  9. 央视公然诽谤Linux操作系统,谁报告?
  10. openstack controller ha测试环境搭建记录(十二)——配置neutron(计算节点)
  11. permutation test
  12. jmeter通过org.sqlite.JDBC驱动连接db数据库
  13. 分布式唯一id:snowflake算法思考
  14. atomic详解
  15. Java开发笔记(六)特殊数字的表达
  16. leecode第一百六十题(相交链表)
  17. Linux路由表信息-route命令
  18. 关于C#中的动态数组ArrayList
  19. 制作签名jar放置到前端资源目录下
  20. centos7默认安装没有连接到网络

热门文章

  1. hive的排序 order by和sort by
  2. 西门子 S7-300 以太网模块连接 WINCC方案
  3. Nextcloud 使用教程
  4. nmap扫描端口导致线上大量Java服务FullGC甚至OOM
  5. 浅谈,seata在使用feign-url通过域名调用时分布式事务不生效的问题及解决
  6. ARM Cortex-M嵌入式C基础编程(下)
  7. MySQL 主从复制&读写分离 简介
  8. .NET平台系列27:在 Linux 上安装 .NET Core/.NET5/.NET6
  9. 使用 Apache Superset 可视化 ClickHouse 数据
  10. [Azure DevOps] 编译时自动修改版本号