上一篇简单讲解了 线程和线程池以及上下文切换。创建线程代价高昂,默认每个线程都要占用大量虚拟内存1M。更有效的做法使用线程池,重复利用线程。在.NET4.0中引入了TPL任务并行库,你可以在将精力集中于程序要完成的工作,同时最大程度地提高代码的性能。在C#5.0中引入了async 和 await关键字,基于任务的异步模式(TAP),所以了解Task对后面学习异步操作会简单些。

任务是封装了以异步方式执行的工作。当启动一个任务,控制几乎立即返回调用者,无论任务要执行多少工作。

创建Task任务

有三种创建方式

  • 使用task构造函数
  • task工厂类静态方法
  • 使用.NET4.5新引入的Task.run()。

我们创建一个输出300万个32位字符的GUID任务分别使用三种不同方式实现。代码如下 constint RepeatCount = ; //重复次数

            var listGuid = new BlockingCollection<string>();

            //Action无返回值
Action dowork = () =>
{
for (var count = ; count < RepeatCount; count++)
{
listGuid.Add(Guid.NewGuid().ToString("N"));
}
};
Task task1 = new Task(dowork); //1)使用构造函数
task1.Start();
Task task2 = Task.Factory.StartNew(dowork); //2)Task工厂方法,直接运行,不需要在调用start()
Task task3 = Task.Run(dowork); //3)4.5 Task.Run 是Task.Factory.StartNew简化方式;直接运行,不需要在调用start() Task.WaitAll(task1, task2, task3); //等待所有任务完成,相当于 thread.join()
Console.Write($"生成数量:{listGuid.Count / 10000}万");

输出

上述实例创建一个没有返回值的任务,当然也可以通过Task<TResult> 来创建返回值的异步操作。

连续任务

第一个任务生成32位字符的Guid任务,利用返回的结果再转化成对应的ASCII码,最后ASCII码十进制的值相加。代码如下

 //Func
Func<string> doWork = () =>
{
return Guid.NewGuid().ToString("N");
};
//延续任务
var task = Task.Run(doWork).ContinueWith(async strGuid =>
{
var resut = await strGuid;
var array = Encoding.ASCII.GetBytes(resut); int mLenght = array.Length;
int sumResult = ;
for (int m = ; m < mLenght; m++)
{
sumResult += array[m];
}
Console.WriteLine($"Guid对应10进制相加结果:{sumResult}");
});

输出

处理任务异常

同步代码要想捕获异常,只需在代码块上添加Try ...Catch即可。但是异步调用不能这么做。因为控制会立即从调用返回,然后控制会离开Try块,而这时距离工作者线程发生异常可能还有好久呢。

为了处理出错的任务,一个技术是显式创建延续任务作为那个任务的“错误处理程序”。检测到先驱任务引发未处理的异常,任务调度器会自动调度延续任务。但是,如果没有这种处理程序,同时在出错的任务上执行wait()(或其他试图获取result的动作),就会引发一个AggregateException,示例代码如下。

  Task task = Task.Run(() =>
{
throw new InvalidOperationException();
}); try
{
task.Wait();
}
catch (Exception ex)
{
Console.WriteLine($"常规erro:{ex.Message};type:{ex.GetType()}");
AggregateException excetion = (AggregateException)ex;
excetion.Handle(eachException =>
{
Console.WriteLine($"erro:{eachException.Message}");
return true;
});
}

输出

虽然工作者线程上已发的未处理异常是InvalidOperationException类型,但主线程捕捉的仍是一个AggregateException。由于编译时不知道工作者任务将要引发一个还是多个异常,所以未处理的出错任务总是引发一个AggregateException。

还可查看任务的Exception属性来了解出错任务的状态,这样不会造成在当前线程上重新引发异常。代码如下

  bool paraentTaskFaulted = false;
Task task = new Task(() =>
{
throw new InvalidOperationException();
}); Task continuationTask = task.ContinueWith(t =>
{ paraentTaskFaulted = t.IsFaulted;
}, TaskContinuationOptions.OnlyOnFaulted); task.Start();
Console.Write(continuationTask.Status);
continuationTask.Wait();
//如果断言失败 则显示一个消息框,其中显示调用堆栈。
Trace.Assert(paraentTaskFaulted);
if (!task.IsFaulted)
{
task.Wait();
}
else
{
task.Exception.Handle(eachException =>
{
Console.WriteLine($"erro:{eachException.Message}");
return true;
});
}

注意,为了获取原始任务上的未处理异常,我们使用Exception属性。结果和上面示例输出一样。

取消任务

任务支持取消,比如常用在指定时间内的任务或者基于某些条件手动的取消,支持取消的任务要监听一个CancellationToken对象。任务轮询它,检查是否出发了取消请求。如下代码展示了取消请求和对请求的响应。

 /// <summary>
/// 取消任务
/// </summary>
public void TaskTopic5()
{
string stars = "*".PadRight(Console.LargestWindowWidth-,'*');
Console.WriteLine("push enter to exit.");
CancellationTokenSource cancellationTokenSource = new CancellationTokenSource(); //向应该被取消的 System.Threading.CancellationToken 发送信号
Task task = Task.Run(
()=>
Count(cancellationTokenSource.Token,),
cancellationTokenSource.Token); Console.Read();
cancellationTokenSource.Cancel();//按下enter键, 传达取消请求 Console.WriteLine(stars);
Console.WriteLine(task.IsCanceled);
task.Wait();
Console.WriteLine();
}
/// <summary>
/// 数数
/// </summary>
/// <param name="token"></param>
/// <param name="countTo"></param>
private void Count(CancellationToken token,int countTo)
{
for (int count = ; count < countTo; count++)
{
//监控是否取消
if (token.IsCancellationRequested)
{
Console.WriteLine("数数喊停了");
break;
} Console.Write(count+"=》");
Thread.Sleep(TimeSpan.FromSeconds());
}
Console.WriteLine("数数结束");
}

输出

调用Cancel()实际会在从cancellationTokenSource.Token复制的所有取消标志上设置IsCancellationRequested属性。

到此任务的一些基本的操作已经完成了,下一节关注下C#5.0的async/await上下文关键字。

对话Task的更多相关文章

  1. 基于人工智能标记语言 (AIML)和任务型对话系统(Task)的深度智能对话机器人demo

    起因 本demo基于基于人工智能标记语言 (AIML)和开放域问答(WebQA)的深度智能对话模型而来 无意间发现一个基于人工智能标记语言 (AIML)和开放域问答(WebQA)的深度智能对话模型,但 ...

  2. 在Dynamics 365中使用SURVEYJS代替对话(Dialog)制作话术

    本人微信和易信公众号: 微软动态CRM专家罗勇 ,回复269或者20180318可方便获取本文,同时可以在第一间得到我发布的最新的博文信息,follow me!我的网站是 www.luoyong.me ...

  3. C#并行开发_Thread/ThreadPool, Task/TaskFactory, Parallel

    大家好,本次讨论的是C#中的并行开发,给力吧,随着并行的概念深入,哥也赶上这个潮流了,其实之前讨论C#的异步调用或者C#中BeginInvoke或者Invoke都已经涉及了部分本篇的内容. 参考书目: ...

  4. 手把手教你利用微软的Bot Framework,LUIS,QnA Maker做一个简单的对话机器人

    最近由于要参加微软亚洲研究院的夏令营,需要利用微软的服务搭建一个对话Bot,以便对俱乐部的情况进行介绍,所以现学了几天,搭建了一个简单的对话Bot,期间参考了大量的资料,尤其是下面的这篇博客: htt ...

  5. 下一个时代,对话即平台 —— 开始使用Bot Framework和Cognitive Service来打造你的智能对话服务

    在16年3月30号微软的全球开发者大会Build上发布了Bot Framework,微软认为下一个big thing是Conversation as a Platform,简称CaaP,中文应该叫做& ...

  6. 学习笔记(5)- ubuntu对话语料

    The Ubuntu Dialogue Corpus: A Large Dataset for Research in Unstructured Multi-Turn Dialogue Systems ...

  7. 一句 Task.Result 就死锁, 这代码还怎么写?

    一:背景 1. 讲故事 前些天把 .NET 高级调试 方面的文章索引到 github 的过程中,发现了一个有意思的评论,详见 文章,截图如下: 大概就是说在 Winform 的主线程下执行 Task. ...

  8. BERT生成能力改进:分离对话生成和对话理解

    NLP论文解读 原创•作者 | 吴雪梦Shinemon 研究方向 | 计算机视觉 导读说明: NLP任务大致可以分为NLU(自然语言理解)和NLG(自然语言生成)两种,NLU负责根据上下文去理解当前用 ...

  9. Concepts:Request 和 Task

    当SQL Server Engine 接收到Session发出的Request时,SQL Server OS将Request和Task绑定,并为Task分配一个Workder.在TSQL Query执 ...

随机推荐

  1. compatible

    compatible 英 [kəmˈpætəbl] 美 [kəmˈpætəbəl] adj. 兼容的,相容的; 和谐的,协调的; [生物学] 亲和的; 可以并存的,能共处的;

  2. jsonarray根据id排序

    List<JSONObject> jsonValue=new ArrayList<JSONObject>(); JSONArray sortJsonarr=new JSONAr ...

  3. centOS7搭建nexus私服

    1.保证JDK,MAVEN已安装,firewalld服务安装 PS:yum install firewalld 2.官网下载:https://www.sonatype.com/download-oss ...

  4. linux resin 基本站点配置

    进入配置文件目录: [root@linuxidc resin-]# cd /usr/local/resin/conf/ 查看都有哪些配置文件: [root@linuxidc conf]# ls app ...

  5. json-server使用及路由配置

    1.先安装node.js,node.js中包含了json-server模块 2.在angular-hello/src/app/data-base.json文件中,编辑json格式的服务数据, { &q ...

  6. python小结 1

    1.变量 记录状态 类型:数字,字符串,元组,列表,字典 可变不可变(内存地址不变的情况下,值能不能改变): 不可变:字符串,数字,元组 可变:列表,字典 访问顺序: 直接访问:数字 有序:字符串,列 ...

  7. jdk8 永久代变更

    java8 去掉了永久代permgen(又称非堆,其实也是堆的一部分),类的方法代码,常亮,方法名,静态变量等存放在永久代中 改为使用元空间 Metaspace , Metaspace 不在是堆的一部 ...

  8. 2019.02.17 spoj Query on a tree V(链分治)

    传送门 题意简述: 给你一棵nnn个黑白点的树,初始全是黑点. 现在支持给一个点换颜色或者求整颗树中离某个点最近的白点跟这个点的距离. 思路: 考虑链分治维护答案,每个链顶用一个堆来维护答案,然后对于 ...

  9. Solutions and Summay for Linked List Naive and Easy Questions

    1.Remove Linked List Elements package linkedlist; /* * Question: Remove all elements from a linked l ...

  10. VS工具栏没有出现可用工具的情况

    (1)没有切换到资源视图,打开具体的对话框. (2)如果你在调试状态,即使打开了具体的对话框,VS工具箱还是不会出现可用的控件的.所以不要在调试状态下添加控件.