c# 基于任务的异步编程模式(TAP)的异常处理

在前面讲到了《基于任务的异步编程模式(TAP)》,但是如果调用异步方法,没有等待,那么调用异步方法的线程中使用传统的try/catch块是不能捕获到异步方法中的异常。因为在异步方法执行出现异常之前,已经执行完毕。

1、没有等待的调用异步方法

ThrowAfter方法是在一定延迟后抛出一个异常:

private async Task ThrowAfter(int ms,string message)
{
  await Task.Delay(ms);
  Console.WriteLine("异步任务随后将抛出异常。");
  throw new Exception(message);
}

DontHandle方法在调用异步方法时,由于有滞后性,所以使用try...catch...不能捕获到异步方法中的异常。

public void DontHandle()
{
  try
  {
    ThrowAfter(200, "异步方法抛出的异常");
  }
  catch(Exception ex)
  {
    Console.WriteLine(ex.Message);
  }
  Console.WriteLine("完成方法:DontHandle");
}

注意:返回void的异步方法不会等待。因为从async void方法抛出的异常无法捕获。因此,异步方法最好返回一个Task类型。

2、异步方法的异常处理

异步方法异常的比较好的处理方式使使用await关键字,将其放在try/catch语句中。

public async void HandleOneError()
{
  Console.WriteLine("HandleOneError方法开始执行。。。");
  try
  {
    await ThrowAfter(2000, "异步方法抛出的异常");
  }
  catch (Exception ex)
  {
    Console.WriteLine(ex.Message);
  }
  Console.WriteLine("完成方法:HandleOneError");
}

调用ThrowAfter方法后,HandleOneError会释放线程,但它会在任务完成时保持对任务的引用。当异步方法抛出异常,会调用匹配的catch块内的代码。

3、多个异步方法的异常处理

如果调用多个异步方法,会有多个抛出异常,在捕获异常时就会有问题。

public async void StartTwoTasks()
{
  Console.WriteLine("StartTwoTasks方法开始执行。。。");
  try
  {
    await ThrowAfter(2000, "first");//先执行该方法
    await ThrowAfter(1000, "Second");//第一个异步方法正常执行完后才会执行该方法
  }
  catch(Exception ex)
  {
    Console.WriteLine(ex.Message);
  }
  Console.WriteLine("完成方法:StartTwoTasks");
}

StartTwoTasks方法中,调用了两个异步方法。理论上认为,当第一个异步方法执行完,抛出异常后,紧接着就会调用第二个异步方法,并抛出异常。但实际上是第一个异步方法抛出异常之后,就会被catch捕获,并不会执行第二个异步方法。因为这种类型中,在“基于任务的异步编程模式(TAP)”一文中解释过,这种调用方法是等待第一个异步方法执行结束后,调用函数的线程控制权才会调用第二个异步方法,多个异步方法以此类推。但是当时我们使用了Task类中的WhenAll方法同时等待多个任务全部执行完,才执行后面的代码。

public async void StartTwoTasksParallel()
{
  Console.WriteLine("StartTwoTasksParallel方法开始执行。。。");
  try
  {
    Task t1 = ThrowAfter(2000, "first");//先执行该方法
    Task t2 = ThrowAfter(1000, "Second");//第一个异步方法执行完后才会执行该方法
    await Task.WhenAll(t1, t2);
  }
  catch (Exception ex)
  {
    Console.WriteLine(ex.Message);
  }
  Console.WriteLine("完成方法:StartTwoTasksParallel");
}

StartTwoTasksParallel方法使用Task类的WhenAll方法,并行调用两个不关联的异步方法。该方法将等待所有任务结束后才结束调用,不论任何一个抛出异常都不会影响其他任务。但是,该方法只会捕获第一个异常(先抛出异常的任务),其他异常将不会被显示。

有一种方法可以获取所有任务的异常信息,就是在try块外声明任务变量t1和t2,让这两个变量在catch块内访问。在catch块中检测任务的IsFaulted属性确认任务的状态,以判定是否出现异常,然后通过Task类的Exception.InnerException访问异常信息本身。

4、使用AggregateException信息

Task.WhenAll方法返回一个Task的结果变量。catch语句只会捕捉到所有异步任务中的第一个异常,但是Task.WhenAll方法返回的Task类型结果变量中会包含所有任务都出现的异常。外部结果任务的Exception属性是一个AggregateException类型,显示所有异常只需要遍历结果任务中的Exception的InnerExceptions属性即可。

public async void ShowAggregatedException()
{
  Console.WriteLine("ShowAggregatedException方法开始执行。。。");
  Task taskResult = null;
  try
  {
    Task t1 = ThrowAfter(2000, "first");//先执行该方法
    Task t2 = ThrowAfter(1000, "second");//第一个异步方法执行完后才会执行该方法
    Task t3 = ThrowAfter(1500, "third");//第一个异步方法执行完后才会执行该方法
    await (taskResult = Task.WhenAll(t1, t2, t3));
  }
  catch (Exception ex)
  {
    Console.WriteLine("handle {0}",ex.Message);
    foreach (Exception ex1 in taskResult.Exception.InnerExceptions)
    {
      Console.WriteLine("Inner exception {0}", ex1.Message);
    }
  }
  Console.WriteLine("完成方法:ShowAggregatedException");
}

以上就是c# 基于任务的异步编程模式(TAP)的异常处理的详细内容,更多关于c# 异步编程的资料请关注我们其它相关文章!

(0)

相关推荐

  • c# 使用异步编程的方法

    怎么使用异步,就是用委托进行处理,如果委托对象在调用列表中只有一个方法,它就可以异步执行这个方法.委托类有两个方法,叫做BeginInvoke和EndInvoke,它们是用来异步执行使用. 异步有三种模式 等待模式,在发起了异步方法以及做了一些其它处理之后,原始线程就中断,并且等待异步方法完成之后再继续. 轮询模式,原始线程定期检查发起的线程是否完成,如果没有则可以继续做一些其它的事情. 回调模式,原始线程一直在执行,无需等待或检查发起的线程是否完成.在发起的线程中的引用方法完成之后,发起的线程

  • C#编程总结(六)详解异步编程

    1.什么是异步? 异步操作通常用于执行完成时间可能较长的任务,如打开大文件.连接远程计算机或查询数据库.异步操作在主应用程序线程以外的线程中执行.应用程序调用方法异步执行某个操作时,应用程序可在异步方法执行其任务时继续执行. 2.同步与异步的区别 同步(Synchronous):在执行某个操作时,应用程序必须等待该操作执行完成后才能继续执行. 异步(Asynchronous):在执行某个操作时,应用程序可在异步操作执行时继续执行.实质:异步操作,启动了新的线程,主线程与方法线程并行执行. 3.异

  • C#实现异步编程的方法

    最近在我参与的几个.Net项目中都有用到异步编程,作为一名.Net小白,很有必要好好地学习一下C#异步编程. 什么是异步 异步指的就是不用阻塞当前线程来等待任务的完成,而是将任务扔到线程池中去执行,当前线程可以继续向下执行,直至其它线程将任务完成,并回调通知当前线程.整个任务从开始到结束都是异步完成的,不会阻塞当前线程.因此,异步很重要的一点就是,不会阻塞当前线程. 实现异步编程 在C#语言中,主要是通过委托来实现异步编程的,在委托类型中定义了两个方法BeginInvoke()和EndInvok

  • C#异步编程详解

    前言 本节主要介绍异步编程中Task.Async和Await的基础知识. 什么是异步? 异步处理不用阻塞当前线程来等待处理完成,而是允许后续操作,直至其它线程将处理完成,并回调通知此线程. 异步和多线程 相同点:避免调用线程阻塞,从而提高软件的可响应性. 不同点: 异步操作无须额外的线程负担,并且使用回调的方式进行处理,在设计良好的情况下,处理函数可以不必使用共享变量(即使无法完全不用,最起码可以减少 共享变量的数量),减少了死锁的可能.C#5.0 .NET4.5 以后关键字Async和Awai

  • 深入分析C#异步编程详解

    同步方法和异步方法的区别同步方法调用在程序继续执行之前需要等待同步方法执行完毕返回结果异步方法则在被调用之后立即返回以便程序在被调用方法完成其任务的同时执行其它操作异步编程概览.NET Framework 允许您异步调用任何方法.定义与您需要调用的方法具有相同签名的委托:公共语言运行库将自动为该委托定义具有适当签名的 BeginInvoke 和 EndInvoke 方法.BeginInvoke 方法用于启动异步调用.它与您需要异步执行的方法具有相同的参数,只不过还有两个额外的参数(将在稍后描述)

  • c# 基于任务的异步编程模式(TAP)

    异步编程是C#5.0的一个重要改进,提供两个关键字:async和await.使用异步编程,方法的调用是在后台运行(通常在线程或任务的帮助下),但不会阻塞调用线程.异步模式分为3种:异步模式.基于事件的异步模式和基于任务的异步模式(TAP).TAP是利用关键字async和await实现的,本文将讲解TAP模式.async和await关键字只是编译器的功能.编译器最终会用Task类创建代码. 1.创建任务 建立一个同步方法Greeting,该方法在等待一段时间后,返回一个字符串. private s

  • C#异步编程几点需要注意的地方

    尽量不要编写返回值类型为void的异步方法 在通常情况下,建议大家不要编写那种返回值类型为void的异步方法,因为这样做会破坏该方法的启动者与方法本身之间的约定,这套约定本来可以确保主调方能够捕获到异步方法所发生的异常. 正常的异步方法是通过它返回的Task对象来汇报异常的.如果执行过程中发生了异常,那么Task对象就进入了faulted(故障)状态.主调方在对异步方法所返回的Task对象做await操作时,该对象若已处在faulted状态,系统则会将执行异步方法的过程中所发生的异常抛出,反之,

  • c# 基于任务的异步编程模式(TAP)的异常处理

    在前面讲到了<基于任务的异步编程模式(TAP)>,但是如果调用异步方法,没有等待,那么调用异步方法的线程中使用传统的try/catch块是不能捕获到异步方法中的异常.因为在异步方法执行出现异常之前,已经执行完毕. 1.没有等待的调用异步方法 ThrowAfter方法是在一定延迟后抛出一个异常: private async Task ThrowAfter(int ms,string message) { await Task.Delay(ms); Console.WriteLine("

  • .NET2.0版本中基于事件的异步编程模式(EAP)

    一.引言 APM为我们实现异步编程提供了一定的支持,同时它也存在着一些明显的问题——不支持对异步操作的取消和没有提供对进度报告的功能,对于有界面的应用程序来说,进度报告和取消操作的支持也是必不可少的. 微软在.NET 2.0的时候就为我们提供了一个新的异步编程模型,也就是基于事件的异步编程模型——EAP(Event-based Asynchronous Pattern ). 二.介绍 实现了基于事件的异步模式的类将具有一个或者多个以Async为后缀的方法和对应的Completed事件,并且这些类

  • C#实现基于任务的异步编程模式

    目录 一.延续任务 二.同步上下文 三.使用多个异步方法 1.按顺序调用异步方法 2.使用组合器 四.转换异步模式 五.错误处理 1.异步方法的异常处理 2.多个异步方法的异常处理 3.使用AggregateException信息返回显示异常 六.取消异步方法 一.延续任务 private async static void CallerWithAsync() { string result = await GreetingAsync("Stephanie"); Console.Wri

  • 基于javascript的异步编程实例详解

    本文实例讲述了基于javascript的异步编程.分享给大家供大家参考,具体如下: 异步函数这个术语有点名不副实,调用一个函数后,程序只在该函数返回后才能继续.JavaScript程序员如果称一个函数为异步的,其意思就是这个函数会导致将来再运行另一个函数,后者取自于事件队列.如果后面这个函数是作为参数传递给前者的,则称其为回调函数. callback 回调函数是异步编程最基本的方式. 采用这种方式,我们把同步操作变成了异步操作,主函数不会堵塞程序运行,相当于先执行程序的主要逻辑,将耗时的操作推迟

  • JavaScript中实现异步编程模式的4种方法

    你可能知道,Javascript语言的执行环境是"单线程"(single thread). 所谓"单线程",就是指一次只能完成一件任务.如果有多个任务,就必须排队,前面一个任务完成,再执行后面一个任务,以此类推. 这种模式的好处是实现起来比较简单,执行环境相对单纯:坏处是只要有一个任务耗时很长,后面的任务都必须排队等着,会拖延整个程序的执行.常见的浏览器无响应(假死),往往就是因为某一段Javascript代码长时间运行(比如死循环),导致整个页面卡在这个地方,其他

  • .NET异步编程模式的三种类型介绍

    一.引言 .NET中很多的类.接口在设计的时候都考虑了多线程问题,简化了多线程程序的开发,不用自己去写WaitHandler等这些底层的代码,由于历史的发展,这些类的接口设计有着三种不同的风格:EAP.APM和TPL.目前重点用TPL. 二.EAP EAP是Event-based Asynchronous Pattem(基于事件的异步模型)的简写,类似于Ajax中的XmlHttpRequest,send之后并不是处理完成了,而是在onreadystatechange事件中再通知处理完成.看下面的

  • 浅谈Async和Await如何简化异步编程(几个实例让你彻底明白)

    引言 C#5.0中async和await两个关键字,这两个关键字简化了异步编程,之所以简化了,还是因为编译器给我们做了更多的工作,下面就具体看看编译器到底在背后帮我们做了哪些复杂的工作的. 同步代码存在的问题 对于同步的代码,大家肯定都不陌生,因为我们平常写的代码大部分都是同步的,然而同步代码却存在一个很严重的问题,例如我们向一个Web服务器发出一个请求时,如果我们发出请求的代码是同步实现的话,这时候我们的应用程序就会处于等待状态,直到收回一个响应信息为止,然而在这个等待的状态,对于用户不能操作

  • .NET中基于事件的异步模式-EAP

    前言 在C# 5.0中,新增了async await 2个关键字支持异步编程的操作.在讲述这两个关键字之前,我先总结一下.NET中的常见的异步编程模型. 异步编程一直是比较复杂的问题,其中要处理多线程之间的数据同步.获取进度.可取消.获取结果.不影响主线程操作.多个任务之间互相不影响等,因此需要设计编程模型去处理此类问题. 从.NET 4.5开始,支持的三种异步编程模式: 基于事件的异步编程设计模式 (EAP,Event-based Asynchronous Pattern) 异步编程模型(AP

  • .NET中的异步编程-EAP/APM使用方法及案例介绍

    从.NET 4.5开始,支持的三种异步编程模式: •基于事件的异步编程设计模式 (EAP,Event-based Asynchronous Pattern) •异步编程模型(APM,Asynchronous Programming Model) •基于任务的编程模型(TAP,Task-based Asynchronous Pattern) 基于任务的异步模式 (TAP) 是基于 System.Threading.Tasks 命名空间的 Task 和 Task<TResult>,用于表示任意异步

随机推荐