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

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

 static void Main(string[] args)
{
RunAsync();
Console.WriteLine("Async Run");
Console.Read();
}
public static void RunAsync()
{
var task = Task.Run(() =>
{
Thread.Sleep();
return "task finished";
});
//当task完成后,会在由线程池继续执行
task.ContinueWith(t =>
{
Thread.Sleep();
Console.WriteLine(task.Result);
});
}

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

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

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

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

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();
Task.Run(() =>
{
//模拟其他任务
Thread.Sleep();
//如果取消请求已发送,则不会显示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传入到任务中,如下

 static void Main(string[] args){
CancelTask();
Console.WriteLine("Async Run");
Console.Read();
}
public static void CancelTask(){
var cancelSource = new CancellationTokenSource();
Task.Run(() =>{
int count = ;
for (int i = ; i < ; i++)
{
if (cancelSource.IsCancellationRequested)
break;
count+=i;
Thread.Sleep();
}
Console.WriteLine(count);
}, cancelSource.Token);
cancelSource.Token.Register(() => Console.WriteLine("Task cancel"));
}

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

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

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

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

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

C#多线程编程(3)--开启子任务的更多相关文章

  1. Java多线程编程实战指南 设计模式 读书笔记

    线程设计模式在按其有助于解决的多线程编程相关的问题可粗略分类如下. 不使用锁的情况下保证线程安全: Immutable Object(不可变对象)模式.Thread Specific Storage( ...

  2. Web Worker javascript多线程编程(二)

    Web Worker javascript多线程编程(一)中提到有两种Web Worker:专用线程dedicated web worker,以及共享线程shared web worker.不过主要讲 ...

  3. 5天玩转C#并行和多线程编程 —— 第三天 认识和使用Task

    5天玩转C#并行和多线程编程系列文章目录 5天玩转C#并行和多线程编程 —— 第一天 认识Parallel 5天玩转C#并行和多线程编程 —— 第二天 并行集合和PLinq 5天玩转C#并行和多线程编 ...

  4. IOS高级编程之三:IOS 多线程编程

    多线程的概念在各个操作系统上都会接触到,windows.Linux.mac os等等这些常用的操作系统,都支持多线程的概念. 当然ios中也不例外,但是线程的运行节点可能是我们平常不太注意的. 例如: ...

  5. Java多线程编程核心技术---Java多线程技能

    基本概念 进程是操作系统结构的基础,是一次程序的执行,是一个程序及其数据结构在处理机上顺序执行时所发生的活动,是程序在一个数据集合上运行的过程,是系统进行资源分配和调度的独立单位.线程可以理解成是在进 ...

  6. iOS开发——多线程篇——NSOperation(基于GCD多线程编程),下载图片并合成新图片

    一.NSOperation的基本概念1.简介NSOperation的作用配合使用NSOperation和NSOperationQueue也能实现多线程编程 NSOperation和NSOperatio ...

  7. 多线程编程1 - NSThread

    每个iOS应用程序都有个专门用来更新显示UI界面.处理用户的触摸事件的主线程,因此不能将其他太耗时的操作放在主线程中执行,不然会造成主线程堵塞(出现卡机现象),带来极坏的用户体验.一般的解决方案就是将 ...

  8. Java多线程编程——进阶篇二

    一.线程的交互 a.线程交互的基础知识 线程交互知识点需要从java.lang.Object的类的三个方法来学习:    void notify()           唤醒在此对象监视器上等待的单个 ...

  9. android: 多线程编程基础

    9.1   服务是什么 服务(Service)是 Android 中实现程序后台运行的解决方案,它非常适合用于去执行那 些不需要和用户交互而且还要求长期运行的任务.服务的运行不依赖于任何用户界面,即使 ...

  10. 深入HTML5 Web Worker应用实践:多线程编程

    HTML5 中工作线程(Web Worker)简介 至 2008 年 W3C 制定出第一个 HTML5 草案开始,HTML5 承载了越来越多崭新的特性和功能.它不但强化了 Web 系统或网页的表现性能 ...

随机推荐

  1. Office 365 机器人(Bot)开发入门指南 (新篇)

    最近在整理书稿时,发现我在2017年7月份写的这篇 Office 365 机器人(Bot)开发入门 的内容,因为相关平台的升级,已经完全不能体现当前的开发过程,所以我再专门写一篇新的开发入门指南给有兴 ...

  2. Selenium_WebDriver_定位元素

    版权声明:本文为博主原创文章,转载请注明出处. 定位单个元素 WebDriver提供了八种元素定位方法,Java中定位语句形如:driver.findElement(By.id()): 何为元素定位? ...

  3. grep 同时满足多个关键字、满足任意关键字和排除关键字

    1. 同时满足多个关键字 grep "word1" file_name | grep "word2" | grep "word3" 2. 满 ...

  4. JDBCTemplate与模板设计方法(二)

    前言:上一篇博客介绍了模板方法模式,并且给出了一个小demo,简单对模板方法进行了实现,接下来我们把目光转向spring的源码JDBCTemplate,看一看spring是如何对jdbc进行高度封装的 ...

  5. nginx/php-fpm 访问php文件直接下载而不运行

    遇到这种问题,首先确认你web服务器配置中的.PHP是不是被指定给FastCGI server处理: location ~ .php$ { fastcgi_pass ; } 如已配置,那么可能是由于f ...

  6. VS2015安装时问题汇总

    安装VS2015遇到teamexplorer严重错误 在控制台管理员权限执行: fsutil behavior set SymlinkEvaluation L2L:1 L2R:1 R2L:1 R2R: ...

  7. 关于ruby -gem无法切换淘宝源

    ruby官网提供的 淘宝的gem源 不起作用 https://ruby.taobao.org/ taobao Gems 源已停止维护,现由 ruby-china 提供镜像服务 http://gems. ...

  8. 分享:Python中的位运算符

    按位运算符是把数字看作二进制来进行计算的.用的不太多,简单了解. 下表中变量 a 为 60,b 为 13二进制格式如下: a = 0011 1100 b = 0000 1101 a&b = 0 ...

  9. 浅谈 Glide - BitmapPool 的存储时机 & 解答 ViewTarget 在同一View显示不同的图片时,总用同一个 Bitmap 引用的原因

    作者:林冠宏 / 指尖下的幽灵 掘金:https://juejin.im/user/587f0dfe128fe100570ce2d8 博客:http://www.cnblogs.com/linguan ...

  10. word自动备份,word误删内容恢复

    有个问题时长困扰着我,就是一次不小心把word里面的一部分内容误删了之后,又手残点击ctrl+s给保存了,要是立即ctrl+z还能撤销,可要是关闭了word才想起来撤销就来不及啦,现在终于找到解决的办 ...