C#多线程编程(3)--开启子任务

上一篇我讲解了await和async关键字,这两个关键字的作用是将async限定的方法中await关键字后面的部分封装成一个委托,该委托会在await修饰的Task完成后再执行。简单的说,就是等待任务完成后,后面的程序才执行,且该等待不会造成线程阻塞。关键是在任务执行完成后,程序会继续交给主线程执行。接下来,我来介绍在任务执行结束后,用新任务来执行方法。

 

废话不多上,上代码,我们来看看如何在任务结束后继续由线程池继续完成其他方法。

 1 static void Main(string[] args)
 2 {
 3     RunAsync();
 4     Console.WriteLine("Async Run");
 5     Console.Read();
 6 }
 7 public static void RunAsync()
 8 {
 9     var task = Task.Run(() =>
10     {
11         Thread.Sleep(2000);
12         return "task finished";
13     });
14     //当task完成后,会在由线程池继续执行
15     task.ContinueWith(t =>
16     {
17         Thread.Sleep(2000);
18         Console.WriteLine(task.Result);
19     });
20 }

可以看到,RunAsync方法中并没有添加async和await关键字,但是运行程序后,主线程没有被阻塞。这是因为ContinueWith会将委托参数“挂”到线程池的任务队列中,该委托只有在task执行结束后,才会开始执行。ContinueWith方法会返回一个Task,该Task就是task执行结束后会开始执行的Task,该Task也是可以等待的。注意,ContinueWIth和await关键字的区别就在:await在任务完成后,会由主线程继续执行,但是ContinueWith中的方法会继续由线程池执行。

接下来,我们来了解一下,任务能否取消,以及取消后会发生什么。

C#提供的任务取消的方式是“协作式”取消,在构造任务时,需要先构造一个System.Threading.CancellationTaskSource对象,该对象看起来像这样

1  public sealed class CancellationTaskSource:IDisposable{
2     public CancellationTokenSource();
3     public void Dispose();//释放资源
4     public bool IsCancellationRequested{get;}
5     public CancellationToken Token{get;}  
6     public void Cancel();//内部调用Cancel并传递false
7     public void Cancel(bool throwOnFirstException);
8 }

CancellationTaskSource对象中含有一个Token,它的类型是CancellationToken,这是一个轻量级值类型,它包含一个私有的对CancellationTaskSource的引用,当构造Task时,需要将这个Token传入,如下所示

static void Main(string[] args){
    CancelTask();
    Console.WriteLine("Async Run");
    Console.Read();
}
public static void CancelTask()
{
    //定时1000ms后自动取消任务
    var cancelSource = new CancellationTokenSource(1000);
    Task.Run(() =>
    {
        //模拟其他任务
        Thread.Sleep(4000);
        //如果取消请求已发送,则不会显示Task over
        if (!cancelSource.IsCancellationRequested)
            Console.WriteLine("Task over");
    }, cancelSource.Token);
    cancelSource.Token.Register(() => Console.WriteLine("Task cancel"));
}

 

运行后可以看到大约1秒后,控制台显示了Task cancel,并且在没有显示Task over。证明任务被取消了。Task.Run()方法的其中一个重载是接受一个CancelToken参数,我们传入的是cancelSource的Token,这表明cancelSource就可以控制Task的取消与否。我的做法是在1秒后自动取消任务,也可以手动调用cancelSource.Cancel()方法来显式取消任务。我在Token上订阅了取消的事件,在任务被取消后,执行我的方法,这里是控制台打印出Task cancel。也可以在任务中,利用“闭包”,将Token传入到任务中,如下

 1 static void Main(string[] args){
 2     CancelTask();
 3     Console.WriteLine("Async Run");
 4     Console.Read();
 5 }
 6 public static void CancelTask(){
 7     var cancelSource = new CancellationTokenSource(1000);
 8     Task.Run(() =>{
 9         int count = 0;
10         for (int i = 0; i < 1000; i++)
11         {
12             if (cancelSource.IsCancellationRequested)
13                 break;
14             count+=i;
15             Thread.Sleep(30);
16         }
17         Console.WriteLine(count);
18     }, cancelSource.Token);
19     cancelSource.Token.Register(() => Console.WriteLine("Task cancel"));
20 }

手动取消有2个方法,一是cancelSource.Cancel(),另一个是cancelSource.CancelAfter(),示例如下

 1 public static void CancelTask(){
 2     var cancelSource = new CancellationTokenSource();
 3     Task.Run(() =>
 4     {
 5         Thread.Sleep(4000);
 6         Console.WriteLine("Task over");
 7     }, cancelSource.Token);
 8     cancelSource.Token.Register(() => Console.WriteLine("Task cancel"));
 9     //此处调用cancelSource.Cancel()的话会立即结束任务
10     //该方法会大约1秒后取消任务
11     cancelSource.CancelAfter(1000);
12 }

该示例和上面的在构造函数中传入1000ms的延时是一样的效果,若调用Cancel(),则立刻结束任务。

以上就是本篇内容,介绍了Task.ContinueWith方法,以及如何取消任务。若文中不能满足你的要求,可以查看诸如Task.Run()或者Task.ContinueWith()方法的几个重载,还有其他的使用方法,本人并没有将全部细枝末节全部学会,主要的目的是掌握多线程的基本使用方法。《CLR via C#》书中也有很详细的讲解,我的这个系列就是从该书中提取,有兴趣的小伙伴可以看一看。欢迎有问题的和我在评论区交流。

下一篇,我给大家介绍一下C#并行的奇技淫巧:并行的For、Foreach循环以及PLINQ。

posted @ 2018-03-08 11:48  JAZzzzzzzz  阅读(761)  评论(7编辑  收藏  举报