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

在.net中,async/await的理解

一、什么是同步?什么是异步?

在.net中,async 和 await 是两个关键字,async 关键字用于声明一个方法是异步方法,该方法可以包含一个或多个 await 表达式。await 关键字是用于在异步方法中等待一个任务(Task 或 Task<T>对象)的完成。在 async 方法中使用 await表达式时,会暂停当前方法的执行,直到等待的任务完成。在这段时间内,主线程可以去执行其它操作。

什么是同步:当一个方法被调用时,调用者需要等待该方法执行完毕后才会继续往下执行,我们称这种方法为同步方法。

什么是异步:当一个方法被调用时立即返回,并获取一个线程执(Task)行该方法内部的业务逻辑,而调用者不需要等待这个方法执行完毕,我们称这种方法为异步方法。

二、async/await是怎么提高性能的?

异步的好处在于非阻塞(调用线程不会暂停执行去等待子线程完成),因此,我们可以把一些不需要立即使用结果、耗时的任务设为异步去执行,可以提高程序的执行效率。

比如,一个主线程需要执行 5 个方法,假设每个方法的分别用时为 0.1s、0.2s、0.3s、0.4s、0.5s,如果在同步编程中,这个主线程的执行用时大概为1.5秒。而如果把这5个方法写成异步的形式,那么这个主线程大概用时为0.5秒。

这是为什么呢?这是因为,在同步方法中,主线程在调用方法时,需要把这个方法执行完成之后再继续调用后续的方法,主线程执行这5个方法就好比一根线穿5颗珠子一样,一颗一颗来,所以主线程执行所用时间大概为1.5秒。而在异步方法中,主线程在调用异步方法时,主线程不会立即去执行异步方法的,而是在遇到异步方法中的 await语句后返回一个任务(Task),然后再继续调用后续的方法,这些任务的执行是由 task创建一个子线程去执行的,主线程执行这5个异步方法就好比5个人拿5根线同时穿5颗珠子,所用时间就是用时最多的那个,所以主线程执行时间大概为0.5秒。

问题:怎么解决在调用异步函数时,主线程继续向下执行后,主线程是怎么回收或管理这个异步方法的执行结果的。

1、await等待执行

public class Demo
{public async Task DemoAsync(){await Task.Delay(1000);Console.WriteLine("1秒后执行");}
}
public Class Program
{static async Task Main(string[] args){Demo demo = new Demo();// 使用 await 等待DemoAsync的执行await demo.DemoAsync();Console.ReadKey();}
}

2、使用事件

internal class Program
{static void Main(string[] args){Demo demo = new Demo();demo.OnEvent += (() =>{Console.WriteLine("事件订阅");});demo.DemoAsync();Console.ReadKey();}
}
public class Demo
{public event Action OnEvent;public async Task DemoAsync(){Console.WriteLine("开始执行");await Task.Delay(1000);Console.WriteLine("1秒后执行");OnEvent?.Invoke();}
}

3、回调函数

internal class Program{static void Main(string[] args){Demo demo = new Demo();demo.DemoAsync(() =>{Console.WriteLine("回调函数");});Console.ReadKey();}
}
public class Demo
{public async Task DemoAsync(Action callback){Console.WriteLine("开始执行");await Task.Delay(1000);callback?.Invoke();Console.WriteLine("1秒后执行");}
}

4、使用异步方法但不等待结果

这种方法知识启动了异步操作而不需要结果,可以简单的调用异步方法而不使用 await。这种方式不会阻塞主线程,也不会处理异步操作的结果。

internal class Program
{static void Main(string[] args){Demo demo = new Demo();demo.DemoAsync();Console.ReadKey();}
}
public class Demo
{public async Task DemoAsync(){Console.WriteLine("开始执行");await Task.Delay(1000);Console.WriteLine("1秒后执行");}
}

5、将异步结果存储在变量中

如果想要在某个时刻获取到异步操作中的结果,可以将异步操作的结果存储在变量中,然后再访问它。

internal class Program
{static void Main(string[] args){Demo demo = new Demo();var demoResult = demo.DemoAsync();// 使用 wait等待异步的完成demoResult.Wait();if (demoResult.IsCompleted){Console.WriteLine("str执行完成");Console.WriteLine(demoResult.Result);}Console.ReadKey();}
}
public class Demo
{public async Task<string> DemoAsync(){Console.WriteLine("开始执行");await Task.Delay(1000);Console.WriteLine("1秒后执行");return "异步返回结果";}
}

三、异步到底解决了什么?到底起到了什么样的作用?

1、提高响应性:使用 async 和 await 可以避免等待长时间运行的操作(如 IO 操作)阻塞主线程,从而提高应用程序的响应性。

2、简化异步操作:async 和 await 使得编写异步代码更接近同步代码的写法,这降低了异步编程的复杂性和出错的概率。

3、优化资源使用:异步操作允许线程在等待任务完成时释放,这样可以为其它任务腾出资源,而不是处于空闲等待状态。

四、在使用异步时的一些问题的解决

1、异步的传递性问题

异步方法的异步效果会在调用链上向上传递的,导致异步方法的调用链上的一些列方法也会被标记为异步。这种情况通常发生在下面场景中:

  1. 调用异步方法:如果一个方法调用了一个异步方法,那么这个方法也需要标记为 async,并使用 await 等待结果。
  2. 返回类型的变化:异步方法通常以 Task 或 Task<T> 类型返回,这意味着调用这些异步方法的其他方法也需要更新其返回类型以匹配 Task 或 Task<T> 。

异步方法的传递性的避免:

1、不使用 await 调用异步函数:如果在调用异步函数时,不需要等待异步函数的返回结果,那么可以不适用 await关键字调用异步函数。

2、使用 Task.Run 来启动异步操作:Task.Run 方法可以用来启动一个新的异步操作,它会提供一个新的任务来异步函数,从而避免了异步的传递性。

3、使用事件或回调来处理异步的结果:如果异步函数需要通知调用者操作已完成,可以使用事件或回调来代替直接的 await 调用。

4、将异步结果存放在变量中:如果需要等待异步操作的结果,但又不想立即等待它,可以将异步任务存储在变量中,然后在需要访问时使用 wait 方法来等待异步的完成。


好记性不然烂笔头,在学习的路上留下点痕迹。希望能给大家带来帮助,也期待你的点赞和讨论。

若有不足之处,还请斧正。

相关文章:

在.net中,async/await的理解

一、什么是同步&#xff1f;什么是异步&#xff1f; 在.net中&#xff0c;async 和 await 是两个关键字&#xff0c;async 关键字用于声明一个方法是异步方法&#xff0c;该方法可以包含一个或多个 await 表达式。await 关键字是用于在异步方法中等待一个任务&#xff08;Task…...

水果识别系统 | BP神经网络水果识别系统,含GUI界面(Matlab)

使用说明 代码下载&#xff1a;BP神经网络水果识别系统&#xff0c;含GUI界面&#xff08;Matlab&#xff09; BP神经网络水果识别系统 一、引言 1.1、研究背景及意义 在当今科技迅速发展的背景下&#xff0c;人工智能技术尤其是在图像识别领域的应用日益广泛。水果识别作为…...

40岁开始学Java:Java中单例模式(Singleton Pattern),适用场景有哪些?

在Java中&#xff0c;单例模式&#xff08;Singleton Pattern&#xff09;用于确保一个类只有一个实例&#xff0c;并提供全局访问点。以下是详细的实现方式、适用场景及注意事项&#xff1a; 一、单例模式的实现方式 1. 饿汉式&#xff08;Eager Initialization&#xff09; …...

李宏毅机器学习课程学习笔记04 | 浅谈机器学习-宝可梦、数码宝贝分类器

文章目录 案例&#xff1a;宝可梦、数码宝贝分类器第一步&#xff1a;需要定义一个含有未知数的function第二步&#xff1a;loss of a function如何Sample Training Examples > 如何抽样可以得到一个较好的结果如何权衡模型的复杂程度 Tradeoff of Model Complexity todo 这…...

C++11中的右值引用和完美转发

C11中的右值引用和完美转发 右值引用 右值引用是 C11 引入的一种新的引用类型&#xff0c;用 && 表示。它主要用于区分左值和右值&#xff0c;并且可以实现移动语义&#xff0c;避免不必要的深拷贝&#xff0c;提高程序的性能。左值通常是可以取地址的表达式&#xf…...

Redis详解(实战 + 面试)

目录 Redis 是单线程的&#xff01;为什么 Redis-Key(操作redis的key命令) String 扩展字符串操作命令 数字增长命令 字符串范围range命令 设置过期时间命令 批量设置值 string设置对象,但最好使用hash来存储对象 组合命令getset,先get然后在set Hash hash命令: h…...

ISP CIE-XYZ色彩空间

1. 颜色匹配实验 1931年&#xff0c;CIE综合了前人实验数据&#xff0c;统一采用700nm&#xff08;红&#xff09;、546.1nm&#xff08;绿&#xff09;、435.8nm&#xff08;蓝&#xff09;​作为标准三原色波长&#xff0c;绘制了色彩匹配函数&#xff0c;如下图。选定这些波…...

【强化学习笔记1】从强化学习的基本概念到近端策略优化(PPO)

好久没有更新了。最近想学习一下强化学习&#xff0c;本系列是李宏毅老师强化学习的课程笔记。 1. Policy-based Model 1.1 Actor 在policy-based model中&#xff0c;主要的目的就是训练一个actor。 对于一个episode&#xff08;例如&#xff0c;玩一局游戏&#xff09;&…...

Deepseek对ChatGPT的冲击?

从测试工程师的视角来看&#xff0c;DeepSeek对ChatGPT的冲击主要体现在**测试场景的垂直化需求与通用模型局限性之间的博弈**。以下从技术适配性、效率优化、风险控制及未来趋势四个维度展开分析&#xff1a; --- ### **一、技术适配性&#xff1a;垂直领域能力决定工具选择…...

STM32中的ADC

目录 一&#xff1a;什么是ADC 二&#xff1a;ADC的用途 三&#xff1a;STM32F103ZET6的ADC 3.1ADC对应的引脚 3.2ADC时钟 3.3ADC的工作模式 ​编辑3.4ADC校准 3.5ADC转换结构和实际电压的换算 四&#xff1a;ADC配置步骤 五&#xff1a;两个重要的函数 一&#xff1a…...

开启AI短剧新纪元!SkyReels-V1/A1双剑合璧!昆仑万维开源首个面向AI短剧的视频生成模型

论文链接&#xff1a;https://arxiv.org/abs/2502.10841 项目链接&#xff1a;https://skyworkai.github.io/skyreels-a1.github.io/ Demo链接&#xff1a;https://www.skyreels.ai/ 开源地址&#xff1a;https://github.com/SkyworkAI/SkyReels-A1 https://github.com/Skywork…...

【uniapp】在UniApp中实现持久化存储:安卓--生成写入数据为jsontxt

在移动应用开发中&#xff0c;数据存储是一个至关重要的环节。对于使用UniApp开发的Android应用来说&#xff0c;缓存&#xff08;Cache&#xff09;是一种常见的数据存储方式&#xff0c;它能够提高应用的性能和用户体验。然而&#xff0c;缓存数据在用户清除缓存或清除应用数…...

大白话React第十一章React 相关的高级特性以及在实际项目中的应用优化

假设我们已经对 React 前端框架的性能和可扩展性评估有了一定了解&#xff0c;接下来的阶段可以深入学习 React 相关的高级特性以及在实际项目中的应用优化&#xff0c;以下是详细介绍及代码示例&#xff1a; 1. React 高级特性的深入学习 1.1 React 并发模式&#xff08;Con…...

java容器 LIst、set、Map

Java容器中的List、Set、Map是核心数据结构&#xff0c;各自适用于不同的场景 一、List&#xff08;有序、可重复&#xff09; List接口代表有序集合&#xff0c;允许元素重复和通过索引访问&#xff0c;主要实现类包括&#xff1a; ArrayList 底层结构&#xff1a;动态数组…...

使用IDEA如何隐藏文件或文件夹

选择file -> settings 选择Editor -> File Types ->Ignored Files and Folders (忽略文件和目录) 点击号就可以指定想要隐藏的文件或文件夹...

DOM HTML:深入理解与高效运用

DOM HTML:深入理解与高效运用 引言 随着互联网的飞速发展,前端技术逐渐成为软件开发中的关键部分。DOM(文档对象模型)和HTML(超文本标记语言)是前端开发中的基石。本文将深入探讨DOM和HTML的概念、特性以及在实际开发中的应用,帮助读者更好地理解和使用这两项技术。 …...

形象生动讲解Linux 虚拟化 I/O

用现实生活的比喻和简单例子来解释 Linux 虚拟化 I/O&#xff0c;就像给朋友讲故事一样。 虚拟化 I/O 要解决什么问题&#xff1f; 想象你有一栋大房子&#xff08;物理服务器&#xff09;&#xff0c;想把它分割成多个小公寓&#xff08;虚拟机&#xff09;出租。每个租客&…...

git从零学起

从事了多年java开发&#xff0c;一直在用svn进行版本控制&#xff0c;如今更换了公司&#xff0c;使用的是git进行版本控制&#xff0c;所以打算记录一下git学习的点滴&#xff0c;和大家一起分享。 百度百科&#xff1a; Git&#xff08;读音为/gɪt/&#xff09;是一个开源…...

汽车低频发射天线介绍

汽车低频PKE天线是基于RFID技术的深度研究及产品开发应用的一种天线&#xff0c;在汽车的智能系统中发挥着重要作用&#xff0c;以下是关于它的详细介绍&#xff1a; 移动管家PKE低频天线结构与原理 结构&#xff1a;产品一般由一个高Q值磁棒天线和一个高压电容组成&#xff…...

【Java分布式】Nacos注册中心

Nacos注册中心 SpringCloudAlibaba 也推出了一个名为 Nacos 的注册中心&#xff0c;相比 Eureka 功能更加丰富&#xff0c;在国内受欢迎程度较高。 官网&#xff1a;https://nacos.io/zh-cn/ 集群 Nacos就将同一机房内的实例划分为一个集群&#xff0c;一个服务可以包含多个集…...

【C++】ImGui:极简化的立即模式GUI开发

如果你是GUI开发的新手&#xff0c;或想试试轻量级、易集成的GUI库&#xff0c;ImGui&#xff08;即时模式图形用户界面&#xff09;是个不错的选择。它以简洁的API、跨平台的兼容性和卓越的性能&#xff0c;受到许多开发者的喜爱。无论是为C项目添加调试界面&#xff0c;还是构…...

5G学习笔记之BWP

我们只会经历一种人生&#xff0c;我们选择的人生。 参考&#xff1a;《5G NR标准》、《5G无线系统指南:如微见著&#xff0c;赋能数字化时代》 目录 1. 概述2. BWP频域位置3. 初始与专用BWP4. 默认BWP5. 切换BWP 1. 概述 在LTE的设计中&#xff0c;默认所有终端均能处理最大2…...

1. 搭建前端+后端开发框架

1. 说明 本篇博客主要介绍网页开发中&#xff0c;搭建前端和后端开发框架的具体步骤&#xff0c;框架中所使用的技术栈如下&#xff1a; 前端&#xff1a;VUE Javascript 后端&#xff1a;Python Flask Mysql 其中MySQL主要用来存储需要的数据&#xff0c;在本文中搭建基本…...

深入浅出:插入排序算法完全解析

1. 什么是插入排序&#xff1f; 插入排序&#xff08;Insertion Sort&#xff09;是一种简单的排序算法&#xff0c;其基本思想与我们整理扑克牌的方式非常相似。我们将扑克牌从第二张开始依次与前面已排序的牌进行比较&#xff0c;将其插入到合适的位置&#xff0c;直到所有牌…...

(十一)基于vue3+mapbox-GL实现模拟高德实时导航轨迹播放

要在 Vue 3 项目中结合 Mapbox GL 实现类似高德地图的实时导航轨迹功能,您可以按照以下步骤进行: 安装依赖: 首先,安装 mapbox-gl 和 @turf/turf 这两个必要的库: npm install mapbox-gl @turf/turf引入 Mapbox GL: 在组件中引入 mapbox-gl 并初始化地图实例: <templ…...

DeepSeek到TinyLSTM的知识蒸馏

一、架构设计与适配 模型结构对比&#xff1a; DeepSeek&#xff08;教师模型&#xff09;&#xff1a;基于Transformer&#xff0c;多头自注意力机制&#xff0c;层数≥12&#xff0c;隐藏层维度≥768TinyLSTM&#xff08;学生模型&#xff09;&#xff1a;单层双向LSTM&#…...

【Transformer模型学习】第三篇:位置编码

文章目录 0. 前言1. 为什么需要位置编码&#xff1f;2. 如何进行位置编码&#xff1f;3. 正弦和余弦位置编码4. 举个例子4.1 参数设置4.2 计算分母项4.3 计算位置编码4.4 位置编码矩阵 5. 相对位置信息6. 改进的位置编码方式——RoPE6.1 RoPE的核心思想6.2 RoPE的优势 7. 总结 …...

微信小程序自定义导航栏实现指南

文章目录 微信小程序自定义导航栏实现指南一、自定义导航栏的需求分析二、代码实现1. WXML 结构2. WXSS 样式样式解析:3. JavaScript 逻辑三、完整代码示例四、注意事项与优化建议五、总结微信小程序自定义导航栏实现指南 在微信小程序开发中,默认的导航栏样式可能无法满足所…...

(十 六)趣学设计模式 之 责任链模式!

目录 一、 啥是责任链模式&#xff1f;二、 为什么要用责任链模式&#xff1f;三、 责任链模式的实现方式四、 责任链模式的优缺点五、 责任链模式的应用场景六、 总结 &#x1f31f;我的其他文章也讲解的比较有趣&#x1f601;&#xff0c;如果喜欢博主的讲解方式&#xff0c;…...

20250225-代码笔记03-class CVRPModel AND other class

文章目录 前言一、class CVRPModel(nn.Module):__init__(self, **model_params)函数功能函数代码 二、class CVRPModel(nn.Module):pre_forward(self, reset_state)函数功能函数代码 三、class CVRPModel(nn.Module):forward(self, state)函数功能函数代码 四、def _get_encodi…...