一起谈.NET技术,.Net 4.0 Parallel 编程(五)Task (中)

本文详细介绍了Task并行处理中的几种常见类型,包括Task延续、嵌套Task(分离与非分离)、以及如何取消Task。通过具体示例展示了如何利用ContinueWith方法实现Task之间的连续执行,如何创建分离和非分离的嵌套Task,以及如何通过CancellationToken来取消Task。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

  在上篇文章中我们看过了如何创建Task,本篇文章就各种类型Task的使用进行说明。

  Task Continuations

  首先我们来看看延续的Task,所谓的延续的Task就是在第一个Task完成后自动启动下一个Task。我们通过ContinueWith方法来创建延续的Task。我们假设有一个接受xml解析的服务,首先从某个地方接受文件,然后解析入库,最后返回回执是否解析正确:


  
[TestMethod]
public void TaskParallelPrint()
{
var ReceiveTask
= new Task(() => ReceiveXml());
var ResolveTask
= ReceiveTask.ContinueWith < bool > ((r) => ResolveXml());
var SendFeedBackTask
= ResolveTask.ContinueWith < string > ((s) => SendFeedBack(s.Result));
ReceiveTask.Start();
Console.WriteLine(SendFeedBackTask.Result);
}

  在每次调用ContinueWith方法时,每次会把上次Task的引用传入进来,以便检测上次Task的状态,比如我们可以使用上次Task的Result属性来获取返回值。上面的代买我们也可以这么写:


  
[TestMethod]
public void TaskParallelPrint()
{
var SendFeedBackTask
= Task.Factory.StartNew(() => ReceiveXml())
.ContinueWith
< bool > (s => ResolveXml())
.ContinueWith
< string > (r => SendFeedBack(r.Result));
Console.WriteLine(SendFeedBackTask.Result);
}

  Detached Nested Tasks

  有些情况下我们需要创建嵌套的Task,嵌套里面又分为分离的和不分离的。其创建的方式很简单,就是在Task的body里面创建一个新的Task。如果新的Task未指定AttachedToParent选项,那么就是分离嵌套的。我们看下面这段代码:


  
var outTask = Task.Factory.StartNew(() =>
{
Console.WriteLine(
" Outer task beginning... " );
var childTask
= Task.Factory.StartNew(() =>
{
Thread.SpinWait(
3000000 );
Console.WriteLine(
" Detached nested task completed. " );
});
});
outTask.Wait();
Console.WriteLine(
" Outer task completed. " );
Console.ReadKey();

  我们可以看到运行结果是:

image  上面的代码中outTask.Wait()表示等待outTask执行完成。

  Child Tasks

  我们将上面的代码加上TaskCreationOptions选项:


  
var outTask = Task.Factory.StartNew(() =>
{
Console.WriteLine(
" Outer task beginning... " );
var childTask
= Task.Factory.StartNew(() =>
{
Thread.SpinWait(
3000000 );
Console.WriteLine(
" Detached nested task completed. " );
},TaskCreationOptions.AttachedToParent);
});
outTask.Wait();
Console.WriteLine(
" Outer task completed. " );
Console.ReadKey();

  看到运行结果:

image  Cancellation Task

  如何取消一个Task呢,我们通过cancellation的tokens来取消一个Task。在很多Task的Body里面包含循环,我们可以在轮询的时候判断IsCancellationRequested属性是否为True,如果是True的话,就可以停止循环以及释放资源,同时抛出OperationCanceledException异常出来。来看一段示例代码:


  
var tokenSource = new CancellationTokenSource();
var token
= tokenSource.Token;
var task
= Task.Factory.StartNew(() =>
{
for (var i = 0 ; i < 10000000 ; i ++ )
{
if (token.IsCancellationRequested)
{
Console.WriteLine(
" Task cacel started... " );
throw new OperationCanceledException(token);
}

}
},token);
token.Register(()
=>
{
Console.WriteLine(
" Canceled " );
});
Console.WriteLine(
" Press enter again to cancel task " );
Console.ReadKey();
tokenSource.Cancel();
try
{
task.Wait();
}
catch (AggregateException e)
{
foreach (var v in e.InnerExceptions)
Console.WriteLine(
" msg: " + v.Message);

}
Console.ReadKey();

  总结

  本篇文章中我们看过了创建各种不同的Task以及如何取消Task,下篇文章中会就异常处理以及Task  Laizy进行说明。

转载于:https://www.cnblogs.com/waw/archive/2011/09/01/2162731.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值