分享

《CLR via C#》读书笔记

 牛人的尾巴 2016-12-13

《CLR via C#》读书笔记-.NET多线程(四)

作者:zlbcdn

协作式取消
协作式取消其英文为: cooperative cancellation model。在26.4节中只是很简单的介绍了通过CancellationTokenSource来终结一个异步操作或长时间执行的同步操作。没有具体的分析和说明为什么要这样用。因为终结一个异步操作的方法有很多,可以使用最简单的truefalse变量结束异步操作。因此本次详细整理CLR的在线程取消的模式。本文参考了MSDN及其他网友的相关资料,具体的引用会在文章的尾端。
从.NET4开始,.NET Framework才为异步或需长时间执行的同步操作提供了协作取消模式。通常使用的有两个“东西“,一个是CancellationTokenSource,另一个是struct:CancellationToken。前者是取消请求的发起者,而后者是消息请求的监听者。就像量子世界中的量子纠缠一样,一个是根据现场的环境做出相应的响应,而另一个会立刻做出反应。CancellationTokenSource与CancellationToken就是这样的一个状态。
协作式取消的使用
协作式取消的使用步骤如下:
1、创建CancellationTokenSource实例
2、使用CancellationTokenSource实例的Token属性,获取CancellationToken,并将其传至Task或线程的相关方法中
3、在task或thread中提供根据CancellationToken.IsCancellationRequested属性值进行判定是否应该停止操作的机制
4、在程序中调用CancellationTokenSource实例的cancel方法
这儿有一篇文章,是使用CancellationTokenSource的具体例子。.Net 4.5中通过CancellationTokenSource实现对超时任务的取消
CancellationTokenSource
1、定义
CancellationTokenSource类的定义如下:

[ComVisibleAttribute(false)]
[HostProtectionAttribute(SecurityAction.LinkDemand, Synchronization = true, 
    ExternalThreading = true)]
public class CancellationTokenSource : IDisposable
  • 1
  • 2
  • 3
  • 4

因本类实现了IDisposable的方法,因此在用完时需调用其dispose方法,或者是使用using
2、CancellationTokenSource与CancellationToken的关系
两者的关系如图所示:
两者的关系
通过这张图,可得出:
1、不同的操作使用相同的CancellationTokenSource实例,就可以达到一次调用取消多个操作的目的。
2、CancellationToken为什么会是struct,而不是类
3、其他说明
1、除了CancellationTokenSource与CancellationToken之外,还有一个OperationCanceledException异常类,这个overload的异常类接受Token作为参数,因此在判断具体异常时,可使用本类
4、代码说明
代码如下:

using System;
using System.Threading;

public class Example
{
   public static void Main()
   {
      // Create the token source.
      CancellationTokenSource cts = new CancellationTokenSource();

      // Pass the token to the cancelable operation.
      ThreadPool.QueueUserWorkItem(new WaitCallback(DoSomeWork), cts.Token);
      Thread.Sleep(2500);

      // Request cancellation.
      cts.Cancel();
      Console.WriteLine("Cancellation set in token source...");
      Thread.Sleep(2500);
      // Cancellation should have happened, so call Dispose.
      cts.Dispose();
   }

   // Thread 2: The listener
   static void DoSomeWork(object obj)
   {
      CancellationToken token = (CancellationToken)obj;

      for (int i = 0; i < 100000; i++) {
         if (token.IsCancellationRequested)
         {
            Console.WriteLine("In iteration {0}, cancellation has been requested...",
                              i + 1);
            // Perform cleanup if necessary.
            //...
            // Terminate the operation.
            break;
         }
         // Simulate some work.
         Thread.SpinWait(500000);
      }
   }
}
// The example displays output like the following:
//       Cancellation set in token source...
//       In iteration 1430, cancellation has been requested...
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45

以上方法使用的系统遗留方式,但是希望停止一个task时,参见如下:How to: Cancel a Task and Its Children
操作取消与对象取消(Operation Cancellation Versus Object Cancellation)
在协作式取消操作中,通常都是在方法中通过判断Token的IsCancellationRequested属性,然后根据这个属性的值对操作(或方法)进行相应的处理。因此,常用的协作式取消模式就是Operation Cancellation。PS.Token的IsCancellationRequested只能被设置一次,即当该属性被设置为true时,其不可能再被设为false,不能重复利用。另外,Token在被“用过”后,不能重复使用该对象。即,CancellationTokenSource对象只能使用一次,若希望重复使用,需要在每次使用时,创建新的对象。
除了操作取消之外,还有另外一种情况,我希望当CancellationTokenSource实例调用cancel方法时,调用某个实例中的某个方法。而这个方法内部没有CancellationToken对象。这个时候可以使用CancellationTokenSource的Register方法。
方法的定义如下:

public CancellationTokenRegistration Register(Action callback)
  • 1

其中Action是.NET内部的自定义的委托,其具体的定义:

public delegate void Action()
  • 1

可使用CancellationToken.Register方法完成对实例中方法的调用。如下有一个例子:

using System;
using System.Threading;

class CancelableObject
{
   public string id;

   public CancelableObject(string id)
   {
      this.id = id;
   }

   public void Cancel() 
   { 
      Console.WriteLine("Object {0} Cancel callback", id);
      // Perform object cancellation here.
   }
}

public class Example
{
   public static void Main()
   {
      CancellationTokenSource cts = new CancellationTokenSource();
      CancellationToken token = cts.Token;

      // User defined Class with its own method for cancellation
      var obj1 = new CancelableObject("1");
      var obj2 = new CancelableObject("2");
      var obj3 = new CancelableObject("3");

      // Register the object's cancel method with the token's
      // cancellation request.
      token.Register(() => obj1.Cancel());
      token.Register(() => obj2.Cancel());
      token.Register(() => obj3.Cancel());

      // Request cancellation on the token.
      cts.Cancel();
      // Call Dispose when we're done with the CancellationTokenSource.
      cts.Dispose();
   }
}
// The example displays the following output:
//       Object 3 Cancel callback
//       Object 2 Cancel callback
//       Object 1 Cancel callback
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47

取消操作的监听与响应方式
在一般情况下,在方法内部使用使用Token.IsCancellationRequested属性判断其值,然后根据其值进行后续操作。这种模式可适应大部分的情况。但是有些情况需要额外的处理方式。
特别是当用户在使用一些外部的library代码时,上面提到的方式可能效果不好,更好的方法就是调用Token的方法 ThrowIfCancellationRequested(),让它抛出异常OperationCanceledException,外部的Library截住异常,然后通过判断异常的Token的相关属性值,再进行相应的处理。
ThrowIfCancellationRequested()的方法相当于:

    if (token.IsCancellationRequested) 
        throw new OperationCanceledException(token);
  • 1
  • 2

因此在使用本方法时,通常的用法是(假设自己正在写的代码会被编译为Library,供其他人调用,则自己写的代码应该是这样的):

if(!token.IsCancellationRequested)
{
    //这儿正常的操作,
    //未被取消时,正常的代码和逻辑操作实现
}else
{
    //代表用户进行了取消操作
    //可以进行一些日志记录
    //注销正在使用的资源
    //然后就需要调用方法
    token.ThrowIfCancellationRequested();
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12

当别人使用Library时,需要在catch块中监听OperationCanceledException异常,代码如下:

try
{
    //调用Library的方法
    library.doSomethingMethod();
}
catch(OperationCanceledException e1)
{
    //捕获这个异常,代表是用户正常取消本操作,因此在这儿需要处理释放资源之类的事情
    xxx.dispose();
}
catch(exception e2)
{
    //其他异常的具体处理方法
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14

以上是处理或写供别人使用的Library或DLL时应该遵循的方法。
在方法内部进行处理相关流程时,对于监听用户是否进行了取消操作,有如下的几种方式:
1.轮询式监听(Listening by Polling)
这种方法是最常用的,也是上面提到的,样例如下:

static void NestedLoops(Rectangle rect, CancellationToken token)
{
   for (int x = 0; x < rect.columns && !token.IsCancellationRequested; x++) {
      for (int y = 0; y < rect.rows; y++) {
         // Simulating work.
         Thread.SpinWait(5000);
         Console.Write("{0},{1} ", x, y);
      }

      // Assume that we know that the inner loop is very fast.
      // Therefore, checking once per row is sufficient.
      //就是下面的这句,通过for循环内部的轮询,去判断IsCancellationRequested属性值,从而去决定做其他的事情
      if (token.IsCancellationRequested) {
         // Cleanup or undo here if necessary...
         Console.WriteLine("\r\nCancelling after row {0}.", x);
         Console.WriteLine("Press any key to exit.");
         // then...
         break;
         // ...or, if using Task:
         //若使用Task时,调用ThrowIfCancellationRequested方法,使其抛出异常
         // token.ThrowIfCancellationRequested();
      }
   }
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24

2.通过回调方法处理取消操作(Listening by Registering a Callback)
在比较复杂的情况下,可以使用register方法,注册或登记取消回调方法。如下所示:

using System;
using System.Net;
using System.Threading;
using System.Threading.Tasks;

class CancelWithCallback
{
   static void Main()
   {
      var cts = new CancellationTokenSource();
      var token = cts.Token;

      // Start cancelable task.
      // 这儿使用了一个Task,Task的使用和具体内容可参见多线程(五)
      Task t = Task.Run( () => {
                    WebClient wc = new WebClient();

                    // Create an event handler to receive the result.
                    wc.DownloadStringCompleted += (obj, e) => {
                               // Check status of WebClient, not external token.
                               if (!e.Cancelled) {
                                  Console.WriteLine("The download has completed:\n");
                                  Console.WriteLine(e.Result + "\n\nPress any key.");
                               }
                               else {
                                  Console.WriteLine("The download was canceled.");
                               }
                    };

                    // Do not initiate download if the external token has already been canceled.
                    // 当没有收到取消消息时,则进行相关的下载。
                    // 并且在初始化时,进行了回调方法的登记,因此,当token收到取消的方法时,则调用wc.CancelAsync()
                    if (!token.IsCancellationRequested) {
                       // Register the callback to a method that can unblock.
                       using (CancellationTokenRegistration ctr = token.Register(() => wc.CancelAsync()))
                       {
                          Console.WriteLine("Starting request\n");
                          wc.DownloadStringAsync(new Uri("http://www."));
                       }
                    }
               }, token);

      Console.WriteLine("Press 'c' to cancel.\n");
      char ch = Console.ReadKey().KeyChar;
      Console.WriteLine();
      if (ch == 'c')
         cts.Cancel();

      Console.WriteLine("Press any key to exit.");
      Console.ReadKey();
      cts.Dispose();
   }
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53

在使用register方法时,有几个注意事项:
1、callback方法尽量要快!不要阻碍线程!因此Cancel方法要等到callback方法结束后才返回
2、callback方法要尽量不要再使用多线程。
3.多对象关联
可通过CancellationTokenSource的CreateLinkedTokenSource方法链接多个对象,从而形成一个新的CancellationTokenSource对象
链接中的任何一个对象使用了cancel方法,这个新的“链式”对象也会被取消。如下:

var cts1=new CancellationTokenSource();
cts1.register(()=>Console.writeline("cts1被取消"));

var cts2=new CancellationTokenSource();
cts2.register(()=>Console.writeline("cts2被取消"));

var linkcts=CancellationTokenSource.CreateLinkedTokenSource(cts1,cts2);
linkcts.register(()=>Console.writeline("LinkCts被取消"));

cts2.cancel();

//其输出结果如下:
//LinkCts被取消
//cts2被取消
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14

写在本节学习最后
1、若自己的程序需要封装为library,供其他人调用,则需要做好两点:1、方法需要接受一个token作为参数;2、需要较好的处理OperationCanceledException异常。
2、本节学习主要是结合:《CLR via C#》、MSDN的官网具体的网址在这儿, 以及网友的相关的文章。

    本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。
    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约

    类似文章 更多