• 多线程笔记-CancellationToken(取消令牌)


    介绍

        为什么需要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 都会被取消

     

     

  • 相关阅读:
    jquery ajax参数详解
    压缩解压函数实现
    WCF 大数据量如何从服务端传到客户端
    [DllImport("kernel32.dll")]
    Oracle数据库使用基础和实例
    Js常用的动态效果
    Js使用正则实现表单验证
    Oracle数据库理论知识
    HTML5,CSS3,JavaScript基础知识与使用
    速读《人月神话》
  • 原文地址:https://www.cnblogs.com/fanfan-90/p/12660996.html
Copyright © 2020-2023  润新知