当前位置: 首页 > news >正文

C# 中对 Task 中的异常进行捕获

以下是在 C# 中对 Task 中的异常进行捕获的几种常见方法:

方法一:使用 try-catch 语句

你可以使用 try-catch 语句来捕获 Task 中的异常,尤其是当你使用 await 关键字等待任务完成时。

using System;
using System.Threading.Tasks;class Program
{static async Task Main(){try{await Task.Run(() =>{// 模拟异常抛出throw new Exception("An error occurred in the task.");});}catch (Exception ex){Console.WriteLine($"Caught exception: {ex.Message}");}}
}

在上述代码中,使用 await 等待 Task.Run 中异步执行的代码。当 Task.Run 中的代码抛出异常时,异常会被 catch 块捕获并处理。

方法二:使用 ContinueWith 并处理 Task.Exception

对于不使用 await 的情况,可以使用 ContinueWith 方法来处理任务完成后的情况,包括异常。

using System;
using System.Threading.Tasks;class Program
{static void Main(){Task.Run(() =>{// 模拟异常抛出throw new Exception("An error occurred in the task.");}).ContinueWith(task =>{if (task.IsFaulted){// 处理异常foreach (var ex in task.Exception.InnerExceptions){Console.WriteLine($"Caught exception: {ex.Message}");}}});// 等待任务完成,这里只是为了防止程序提前退出,在实际应用中可能有不同的等待方式Console.ReadLine();}
}

在这个例子中,创建一个 Task 并使用 ContinueWith 来添加后续操作。如果任务发生故障(即抛出异常),task.IsFaulted 将为 true,并且可以通过 task.Exception.InnerExceptions 来访问异常信息。需要注意的是,Task.Exception 是一个 AggregateException,因为一个 Task 可能会抛出多个异常,所以它包含了一个内部异常列表。

方法三:使用 async-awaitTask.WhenAllTask.WhenAny 时的异常处理

当使用 Task.WhenAllTask.WhenAny 组合多个任务时,也可以使用 try-catch 来捕获异常。

using System;
using System.Threading.Tasks;class Program
{static async Task Main(){Task task1 = Task.Run(() =>{throw new Exception("Error in task 1");});Task task2 = Task.Run(() =>{// 正常执行});try{await Task.WhenAll(task1, task2);}catch (Exception ex){Console.WriteLine($"Caught exception: {ex.Message}");}}
}

在这个例子中,Task.WhenAll 会等待 task1task2 都完成。如果 task1 抛出异常,该异常将被 catch 块捕获。对于 Task.WhenAny,异常处理的逻辑类似,但它只会等待第一个任务完成,所以你可能需要额外的逻辑来确保在异常发生后处理其他任务。

方法四:使用 UnobservedTaskException 事件(不推荐)

在一些情况下,可以使用 TaskScheduler.UnobservedTaskException 事件来处理未观察到的异常,但这种方式在.NET 4.5 及以上版本中已经不推荐使用,因为异常可能会导致应用程序崩溃。

using System;
using System.Threading.Tasks;class Program
{static void Main(){TaskScheduler.UnobservedTaskException += (sender, e) =>{foreach (var ex in e.Exception.InnerExceptions){Console.WriteLine($"Unobserved exception: {ex.Message}");}// 标记为已观察,避免程序崩溃e.SetObserved();};Task.Run(() =>{throw new Exception("This is an unobserved exception.");});// 给任务一些时间抛出异常System.Threading.Thread.Sleep(1000);}
}

总结

  • 推荐使用 try-catchawait:对于使用 async-await 模式的异步编程,这是最简洁和直观的方式,能够直接捕获在任务执行过程中抛出的异常。
  • 使用 ContinueWith 进行链式处理:对于不使用 await 的情况,使用 ContinueWith 可以方便地在任务完成后检查是否发生故障并处理异常。
  • 处理多个任务的异常:使用 Task.WhenAllTask.WhenAny 时,仍然可以使用 try-catch 来捕获组合任务中可能出现的异常。

在实际开发中,选择合适的异常处理方法取决于你的具体代码结构和异步编程的使用方式。但总体而言,使用 try-catchawait 结合是最符合现代 C# 异步编程习惯和最安全的方式,能确保异常被妥善处理,避免程序因未处理的异常而崩溃。

http://www.lryc.cn/news/521000.html

相关文章:

  • Android车机DIY开发之软件篇(九)默认应用和服务修改
  • SimpleFOC01|基于STM32F103+CubeMX,移植核心的common代码
  • web.xml常用配置
  • 代码随想录刷题day07|(数组篇)58.区间和
  • 【Linux】进程结束和进程等待
  • 可编辑精品PPT | 城投集团(行业)数字化解决方案
  • 统计学习算法——决策树
  • 基于网络爬虫技术的网络新闻分析
  • 51_Lua面向对象编程
  • 关于在 Kotlin DSL 中,ndk 的配置方式
  • 【论文阅读+复现】High-fidelity Person-centric Subject-to-Image Synthesis
  • Spring Boot 应用开发入门
  • 【C语言】字符串函数详解
  • 【Vim Masterclass 笔记14】S07L29 + L30:练习课08 —— Vim 文本对象同步练习(含点评课内容)
  • 非PHP开源内容管理系统(CMS)一览
  • WEB 攻防-通用漏-XSS 跨站脚本攻击-反射型/存储型/DOMBEEF-XSS
  • SQLAlchemy -批量插入时忽略重复
  • 1月13日学习
  • Steam个人开发者注册备记
  • django在线考试系统
  • Laravel 中 Cache::remember 的基本用途
  • 前端进程和线程及介绍
  • OpenGL —— 基于Qt的视频播放器 - ffmpeg硬解码,QOpenGL渲染yuv420p或nv12视频(附源码)
  • Vue Router
  • 【黑灰产】人工查档业务产业链
  • 114周二复盘 (178)
  • day10_Structured Steaming
  • Python的秘密基地--[章节11] Python 性能优化与多线程编程
  • drawDB docker部属
  • 探索图像编辑的无限可能——Adobe Photoshop全解析