【C#】并行编程实战:基于任务的异步编程基础(下)

news2024/10/3 2:23:45

        第八章介绍了 C# 中可用异步编程的实践和解决方案,还讨论了何时适合使用异步编程等。本章主要介绍 async 和 await 关键字。

        其实在之前的学习中,大家都已经了解过这两个关键字了,用得非常多。其实我觉得没有必要再赘述了,不过这里还是简单地看一看吧。

        本教程学习工程:魔术师Dix / HandsOnParallelProgramming · GitCode

        因为篇幅限制,本篇为下篇,主要内容为异步代码的异常处理和使用异步代码的一些注意事项。


3、异步代码的异常处理

        在使用同步代码的情况下,所有异常都将传播到堆栈顶部,直到它们被 try-catch 块处理或作为未处理的异常抛出为止。

        在使用异步方法时,调用堆栈是不一样的,因为线程先是从方法转换到线程池,然后又回到了线程池。

        接下来将通过示例来了解程序异常的表现:

        public static async Task ExceptionFunction()
        {
            Debug.Log($"开始执行异常方法!");
            await Task.Delay(1500);
            int x = 5;
            Debug.Log($"准备抛出异常!");
            Debug.Log(x / 0);
        }

        这里我们写一个错误方法,最后应该会出错,这里我们直接使用 Task.Run 来运行:

Task.Run(ExceptionFunction);

        结果显而易见,不会有异常的打印,代码直接在错误的地方中断了,而没有下文:

         使用 Thread 来调用上述代码也是一样的,不会抛出异常。

3.1、在 try-catch 外部创建任务

        示例代码如下:

        private void RunWithExceptionFuntion()
        {
            Task task = Task.Run(ExceptionFunction);
            try
            {
                task.ContinueWith(x =>
                {
                    Debug.Log(x.IsFaulted);
                });
            }
            catch (Exception ex)
            {
                Debug.LogError(ex.InnerException);//这样不会抛出异常
            }
        }

        这么写不会有异常抛出:

         try-catch 里没有对此错误有任何感知,就像从来没有发生过一样。 

3.2、在 try-catch 内部创建任务

        现在我们修改一下代码,在将 Task 的创建移动到 try-catch 内部:

        private void RunWithExceptionFuntion()
        {
            try
            {
                Task task = Task.Run(TestFunction.ExceptionFunction);
                task.ContinueWith(x =>
                {
                    Debug.Log(x.IsFaulted);
                });
            }
            catch (Exception ex)
            {
                Debug.LogError(ex.InnerException);
            }
        }

        与书上不一样的是,这样依然没有任何有异常抛出,执行结果和 3.1 一样。在第二章的时候已经说过这个问题了,需要将 Task 调回主线程(Task.Wait),才能正常执行报错。

3.3、用子线程收集子线程的异常

        但是直接调用 Task.Wait 会导致调用线程卡死,如果是主线程调用就会导致主线程卡死以等待异步。显然我们也不想调用线程等待,不然异步就没意义了。

        所以解决方案是:当我们需要启动子线程(Task B)时,先启动一个线程(Task A)来运行 Task B;之后 Task A等待 Task B 完成,以收集错误:
 

        private void RunWithExceptionFuntion()
        {
            RunExceptionFuntionAsync();
        }

        private async void RunExceptionFuntionAsync()
        {
            try
            {
                Task task = Task.Run(TestFunction.ExceptionFunction);
                await task;
                Debug.Log(task.IsFaulted);
            }
            catch (Exception ex)
            {
                Debug.LogError("出错了!");
                Debug.LogError(ex.Message);
            }
        }

简单写法就如上所示,这样就能正常打印出错误了。

3.4、返回 void 会导致程序崩溃?

        在书中提到这么一个案例:

        private void RunWithCrashFunction()
        {
            Task task = TestFunction.ExceptionFunction();
            Debug.Log($"RunWithCrashFunction End {task}");
        }

        如果上述代码返回的是 void 而不是 Task,则会导致程序崩溃。但实际上这个问题并不存在,在我的版本中根本过不了编译:

         这个语法在 ExceptionFunction ,返回值无论是 void 还是 Task 都不会报错,但是在调用时还是严格区分了 void 和 Task 的。所以即便返回值错误,IDE 也会自动帮我们检测出来。

        书上可能是老版本的 C# 了,现在的版本并不存在这种情况。

4、使用 PLINQ 实现异步

        其实使用 PLINQ 在 第四章:使用PLINQ 里已经详细讲解过了,这里就不赘述了。书上这一章节也没有讲什么新东西,可以跳过。

5、衡量异步代码的性能

        这里这本书先讲了一个示例:

        private async void RunWithWaitDebugAll()
        {
            int x = await Utils.WaitWithTask(1100);
            int y = await Utils.WaitWithTask(2100);
            int z = await Utils.WaitWithTask(1500);
            Debug.Log($"返回结果:{x + y + z}");
        }

        像这样的代码,虽然不会阻塞主线程,但是是串行的,任务是完成等待另一个依次执行。书上建议做如下行改造,将三个任务并行:

        private async void RunWithWaitDebugAll()
        {
            int[] ret = await Task.WhenAll(Utils.WaitWithTask(1000), Utils.WaitWithTask(1500), Utils.WaitWithTask(412));
            int sum = 0;
            foreach (var item in ret)
                sum += item;
            Debug.Log($"返回结果:{sum}");
        }

        个运行效果非常明显,同样是不会阻塞主线程,但是等待时间不是 3 个Task 之和,而是其中的最大值。显然在不少使用场景中,这样的写法是有优势的。当然,如果在业务上,确实需要串行以保证逻辑,还是可以使用第一种写法。

        简单地说,就是小任务尽量异步同时派发,用其他方法(例如信号灯、事件)来保证逻辑时序。

6、使用异步代码的准则

        这一章其实并不是我们想象中的一些干货、法则之类的,主要讲了以下事项:

        避免使用异步 void :在前面的章节看到了返回 void 的一些弊端,比如不能等待,或者无任务状态返回等。所以基本上能返回 Task 就返回 Task,这个其实也没什么好说的。

        在 ASP.NET 中会有一些死锁的问题:

Task.ConfigureAwait 方法 (System.Threading.Tasks) | Microsoft Learn配置用于等待此 Task的 awaiter。 icon-default.png?t=N6B9https://learn.microsoft.com/zh-cn/dotnet/api/system.threading.tasks.task.configureawait?view=netstandard-2.1

        但是在 Unity 中使用 Task.Wait 并不会有这些问题。


7、本章小结

        本章感觉偏水……

        无非就是讨论了 async 和 await 两个关键字,然后在提出了 void 和 Task 的返回值之类的问题。但实际上这些问题在之前的章节学习中就已经涉及过了,还写了不少代码测试……

        最后还有就是说,尽量避免写需要切换任务上下文的代码,能并行就并行。当然这个还是根据业务需求来,并在实际工作中实践就好。

         本教程学习工程:魔术师Dix / HandsOnParallelProgramming · GitCode

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/787849.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Git简介与工作原理:了解Git的基本概念、版本控制系统和分布式版本控制的工作原理

🌷🍁 博主 libin9iOak带您 Go to New World.✨🍁 🦄 个人主页——libin9iOak的博客🎐 🐳 《面试题大全》 文章图文并茂🦕生动形象🦖简单易学!欢迎大家来踩踩~&#x1f33…

Centos安装postgresql数据库以及postgis扩展的安装

这几天项目上的数据库迁移差点没把我弄死 ,因为项目上的使用的是postgis来处理地理数据 ,而开发环境的数据库以及postgis扩展并不是我安装的。所以在迁移的时候造成了不小的麻烦。记录一下迁移过程中遇到的。下面以Centos操作系统为例 文章目录 卸载post…

微信小程序——实现手机振动效果

✅作者简介:2022年博客新星 第八。热爱国学的Java后端开发者,修心和技术同步精进。 🍎个人主页:Java Fans的博客 🍊个人信条:不迁怒,不贰过。小知识,大智慧。 💞当前专栏…

解决行业反复“造轮子”现象,全新地平线RDK系列机器人开发者套件上线

7月25日,“地平线2023机器人开发者创享日”在深圳举办,地平线RDK系列机器人开发者套件正式上线,机器人操作系统TogetheROS.Bot™2.0版发布,应用中心NodeHub首发亮相,地平线开发者社区改版上线。 地平线2023机器人开发者…

go性能分析工具之trace

参考文章: https://eddycjy.gitbook.io/golang/di-9-ke-gong-ju/go-tool-trace https://mp.weixin.qq.com/s__bizMzUxMDQxMDMyNg&mid2247484297&idx1&sn7a01fa4f454189fc3ccdb32a6e0d6897&scene21#wechat_redirect 你有没有考虑过,你的g…

HTTP1、 HTTP2、HTTP3 区别

HTTP1、 HTTP2、HTTP3 区别 HTTP1HTTP/1.0和HTTP/1.1的区别HTTP1的缺陷 HTTP2SPDY协议HTTP/2 新特性(即HTTP/2.0和HTTP/1.x的区别)HTTP/2 的缺点 HTTP3QUIC协议HTTP/3 新特性(HTTP/3与HTTP/2、HTTP/1的区别)基于QUIC的0RTT是如何实…

【Linux后端服务器开发】HTTP协议

目录 一、HTTP协议概述 二、HTTP应用层服务器实现 Util.hpp Protocal.hpp Http_Server.hpp http_server.cc indext.html 一、HTTP协议概述 请求和响应怎么保证应用层完整读取完毕了? 读取完整的一行(识别行分隔符),while&…

MobPush Android SDK 厂商推送限制

概述 厂商推送限制 每个厂商通道都有对应的厂商配额和 QPS 限制,当请求超过限制且已配置厂商回执时,MobPush会采取以下措施: 当开发者推送请求超过厂商配额时,MobPush将通过自有通道进行消息下发。当开发者推送请求超过厂商 QP…

elementui el-table 封装表格

ps: 1.3版本 案例&#xff1a; 完整代码&#xff1a; 可直接复制粘贴&#xff0c;但一定要全看完&#xff01; v-slot"scopeRows" 是vue3的写法&#xff1b; vue2是 slot-scope"scope" <template><!-- 简单表格、多层表头、页码、没有合并列行…

2023年度盘点:网络电视盒子哪个好?目前性能最好的电视盒子

挑选电视盒子时配置性能是最重要的&#xff0c;芯片、运存、存储全部会影响到我们的观影体验&#xff0c;想播放流畅、响应速度快&#xff0c;要在预算范围内选择配置最高的产品&#xff0c;那么哪些电视盒子称得上是目前性能最好的电视盒子呢&#xff1f;不懂网络电视盒子哪个…

自定义view - 玩转字体变色

自定义View步骤&#xff1a; 1>&#xff1a;values__attrs.xml&#xff0c;定义自定义属性&#xff1b; 2>&#xff1a;在第三个构造方法中获取自定义属性&#xff1b; 3>&#xff1a;onMeasure【不是必须的】&#xff1b; 4>&#xff1a;onDraw&#xff1a;绘制代…

【Java中的Thread线程的简单方法介绍和使用详细分析】

文章目录 前言一、run() 和 start() 方法二、sleep() 方法三、join() 方法总结 前言 提示&#xff1a;若对Thread没有基本的了解&#xff0c;可以先阅读以下文章&#xff0c;同时部分的方法已经在如下两篇文章中介绍过了&#xff0c;本文不再重复介绍&#xff01;&#xff01;…

海量遥感数据处理与云计算技术教程

详情点击链接&#xff1a;海量遥感数据处理与GEE云计算技术 一&#xff0c;GEE及开发平台 1.GEE平台及典型应用&#xff1b; 2.GEE JavaScript开发环境及常用数据资源&#xff1b; 3.JavaScript基础&#xff0c;包括变量&#xff0c;运算符&#xff0c;数组&#xff0c;判断及…

【数据结构和算法15】二叉树的实现

二叉树是这么一种树状结构&#xff1a;每个节点最多有两个孩子&#xff0c;左孩子和右孩子 重要的二叉树结构 完全二叉树&#xff08;complete binary tree&#xff09;是一种二叉树结构&#xff0c;除最后一层以外&#xff0c;每一层都必须填满&#xff0c;填充时要遵从先左后…

程序员千万别碰这3种副业!!!

最近&#xff0c;“消费降级”这个词频频被大家提及&#xff0c;某瓣上&#xff0c;“今天消费降级了吗”小组的常驻人口有36万&#xff0c;某书上&#xff0c;跟消费降级有关的笔记近7万条...... 不少网友晒出了自己消费降级后的生活&#xff1a;由从前每天一杯的星巴克变成了…

ROS noetic,ROS melodic 安装 MoveIt 并运行

ROS noetic&#xff0c;ROS melodic 安装 MoveIt 并运行 前言更新功能包版本下载依赖文件创建工作区和软件源下载源代码安装编译器缓存&#xff08;可选环节&#xff09;编译Moveit&#xff01;安装Moveit&#xff01;检测是否安装成功 前言 在安装过程中我也碰壁过很多次&…

[算法通关村] 1.3 链表的删除

上一节我们谈到了链表的头插、尾插、中间插入的方法&#xff0c;忘记的小伙伴可以复习一下&#xff1a; [算法通关村] 1.2 链表的插入 接下来&#xff0c;完成链表的删除工作&#xff0c;我们在上一节的学习中&#xff0c;分别在链表的开头、中间和结尾插入了节点&#xff0c;…

【USACO OPEN12铜组】岛屿

【USACO OPEN12铜组】岛屿 文章目录 【USACO OPEN12铜组】岛屿题目描述输入格式输出格式数据范围输入样例&#xff1a;输出样例&#xff1a; 思路code 2014. 岛 - AcWing题库 题目描述 每当下雨时&#xff0c;农夫约翰的田地总是被洪水淹没。 由于田地不是完全水平的&#xf…

Cesium态势标绘专题-扇形(标绘+编辑)

标绘专题介绍:态势标绘专题介绍_总要学点什么的博客-CSDN博客 入口文件:Cesium态势标绘专题-入口_总要学点什么的博客-CSDN博客 辅助文件:Cesium态势标绘专题-辅助文件_总要学点什么的博客-CSDN博客 本专题没有废话,只有代码,代码中涉及到的引入文件方法,从上面三个链…

(五)FLUX中的数据类型

以下内容来自 尚硅谷&#xff0c;写这一系列的文章&#xff0c;主要是为了方便后续自己的查看&#xff0c;不用带着个PDF找来找去的&#xff0c;太麻烦&#xff01; 第 5 章 FLUX中的数据类型 5.1 10 个基本数据类型 5.1.1 Boolean &#xff08;布尔型&#xff09; 5.1.1.1 …