微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

c# – TPL继续任务:不确定谁是父任务

我有Continuationoption.AttachedToParent标志的麻烦.

这是我的伪代码

Task parentTask = Task.Start(() =>   
   { 
        Task childTask = Task.Start(() => doSomething(),Continuationoption.AttachedToParent);
        childTask.ContinueWith(() => followingMethod(),Continuationoption.AttachedToParent);
   }

我知道如果“doSomething()”抛出并且异常childTask失败并且parentTask也因为失败而失败
Continuationoption.AttachedToParent选项.

如果followingMethod()抛出异常,但parentTask状态为Completed,我会期望相同的行为.

我做错了还是继续任务的“父”任务不是我的“parentTask”?

解决方法

你的直觉是正确的,但我猜你最有可能只使用错误的API.

我知道它只是伪代码,但Task.Start()是一个实例方法,而不是一个静态方法,所以不会按照你指示的方式编译,我们不知道你是如何实际启动的任务,细节都很重要. TaskFactory.StartNew会做你想要的,但Task.Run不会.试试这个:

Task parent = Task.Factory.StartNew(() =>
{
  Task child = Task.Factory.StartNew(
     () => { Console.WriteLine("foo"); },TaskCreationoptions.AttachedToParent);
  Task continuation = child.ContinueWith(
     (Task prev) => { throw new InvalidOperationException("Test"); },TaskContinuationoptions.AttachedToParent);
});
try
{
  parent.Wait();
}
catch (AggregateException ex)
{
  if (ex.Flatten().InnerException is InvalidOperationException)
  { 
    Console.WriteLine("The continuation exception was propagated to parent");
  }
}

如果你改变了

Task parent = Task.Factory.StartNew(...)

Task parent = Task.Run(...)

然后你不会得到你想要的行为,因为Task.Run本质上是TaskFactory.StartNew的包装器,其中(除其他外)指定了TaskCreationoptions.DenyChildAttach,因此两个嵌套任务都将分离运行(没有父项).我猜这就是你的问题所在.

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 [email protected] 举报,一经查实,本站将立刻删除。

相关推荐