前面介绍了Task的由来,以及简单的使用,包括开启任务,处理任务的超时、异常、取消、以及如果获取任务的返回值,在回去返回值之后,立即唤起新的线程处理返回值、且如果前面的任务发生异常,唤起任务如果有效的处理异常等关于Task的知识。所以本文将介绍Task更多的用法和特性.
一、如果通过一个任务创建多个子任务.
1、Task支持一个任务,创建多个子任务,并且保持关联.
static void Main(string[] args) { var parentTask = new Task<int[]>(() => { //开启多个子任务 var results = new int[2]; //创建子任务,并将子任务的值赋给results变量,并通过TaskCreationOptions.AttachedToParent,将其关联到父任务,如果不指定,该任务将独立于父任务单独执行 //这里有个奇怪的问题,只能使用new Task的方式去创建关联到父任务的子任务,因为Task.Run没有提供这个方法,可以通过扩展方法解决这个问题 new Task(() => results[0] = ChildThreadOne(), TaskCreationOptions.AttachedToParent).Start(); new Task(() => results[1] = ChildThreadTwo(), TaskCreationOptions.AttachedToParent).Start(); return results; }); parentTask.Start(); parentTask.ContinueWith(x => { Console.WriteLine("当父任务执行完毕时,CLR会唤起一个新线程,将父任务的返回值(子任务的返回值)输出,所以这里不会有任何的线程发生阻塞"); foreach (var re in parentTask.Result) { Console.WriteLine("子任务的返回值分别为:{0}", re); } }); Console.WriteLine("主线程不会阻塞,它会继续执行"); Console.ReadKey();//必须加这行代码,因为Task时线程池线程,属于后台线程 } /// <summary> /// 子任务一 /// </summary> static int ChildThreadOne() { Thread.Sleep(2000);//模拟长时间计算操作 Console.WriteLine("子任务一完成了计算任务,并返回值:{0}", 6); return 6; } /// <summary> /// 子任务一 /// </summary> static int ChildThreadTwo() { Thread.Sleep(2000);//模拟长时间计算操作 Console.WriteLine("子任务二完成了计算任务,并返回值:{0}", 6); return 6; }
二、关于Task的资源释放问题.
如果你看过Task的源码,你会发现下面这个有趣的问题:
ok,你会想它想释放什么呢?
没错,当Task任务,指定了TaskContinuationOptions枚举状态,且指定的值如下:
那么,直接return,什么资源释放操作都不做.
如果任务没有完成,就调用Dispose方法,那么直接抛异常,如果完成了,它就释放了ManualResetEventSlim信号量(后面的文章会介绍).所以如果你在task中使用了其它的一些非托管资源,那么最好在代码里自己手动释放,在使用完之后。或者自己实现了Task的派生类,把需要用的非托管资源加进去,然后在使用完派生类之后,调用Dispose方法.
三、关于Task的几个常用属性
1、Id属性,每个Task对象都有一个Id属性,全局唯一,且每次创建新的任务,这个值都会递增1.
2、TaskStatus状态
// // 摘要: // 表示 System.Threading.Tasks.Task 的生命周期中的当前阶段。 public enum TaskStatus { // // 摘要: // 该任务已初始化,但尚未被计划。 Created = 0, // // 摘要: // 该任务正在等待 .NET Framework 基础结构在内部将其激活并进行计划。 WaitingForActivation = 1, // // 摘要: // 该任务已被计划执行,但尚未开始执行。 WaitingToRun = 2, // // 摘要: // 该任务正在运行,但尚未完成。 Running = 3, // // 摘要: // 该任务已完成执行,正在隐式等待附加的子任务完成。 WaitingForChildrenToComplete = 4, // // 摘要: // 已成功完成执行的任务。 RanToCompletion = 5, // // 摘要: // 该任务已通过对其自身的 CancellationToken 引发 OperationCanceledException 对取消进行了确认,此时该标记处于已发送信号状态;或者在该任务开始执行之前,已向该任务的 // CancellationToken 发出了信号。 有关详细信息,请参阅任务取消。 Canceled = 6, // // 摘要: // 由于未处理异常的原因而完成的任务。 Faulted = 7 }
构造完Task对象是,状态为Created,当任务启动时,状态变为WaitingToRun,当Task实际在线程上运行时,状态变为Running.如果当前任务为父任务,且它已经执行完毕,等待其它子任务执行完毕的时候,其状态变为WaitingForChildrenToComplete.如果任务完成可能会出现以下几种状态:RanToCompletion(已成功完成执行的任务)、Canceled(取消状态)、
Faulted(任务出错).
这里需要注意一个特殊的状态WaitingForActivation
当使用Task对象的ContinueWith的Task对象处理改状态,意味者该Task任务的调度由任务基础结构控制.也就是该任务的调度只有当前面的任务执行完之后,由CLR发起执行调用.