c# async/await的用法

article/2025/9/13 21:01:58

转于:https://www.cnblogs.com/liqingwen/p/5831951.html

关键:

 异步方法:在执行完成前立即返回调用方法,在调用方法继续执行的过程中完成任务。

 async/await 结构可分成三部分:

     (1)调用方法:该方法调用异步方法,然后在异步方法执行其任务的时候继续执行;

     (2)异步方法:该方法异步执行工作,然后立刻返回到调用方法;

     (3)await 表达式:用于异步方法内部,指出需要异步执行的任务。一个异步方法可以包含多个 await 表达式(不存在 await 表达式的话 IDE 会发出警告)。

一、What's 异步?

     启动程序时,系统会在内存中创建一个新的进程。进程是构成运行程序资源的集合。

     在进程内部,有称为线程的内核对象,它代表的是真正的执行程序。系统会在 Main 方法的第一行语句就开始线程的执行。

 

     线程:

     ①默认情况,一个进程只包含一个线程,从程序的开始到执行结束;

     ②线程可以派生自其它线程,所以一个进程可以包含不同状态的多个线程,来执行程序的不同部分;

     ③一个进程中的多个线程,将共享该进程的资源;

     ④系统为处理器执行所规划的单元是线程,而非进程。

 

     一般来说我们写的控制台程序都只使用了一个线程,从第一条语句按顺序执行到最后一条。但在很多的情况下,这种简单的模型会在性能或用户体验上不好。

     例如:服务器要同时处理来自多个客户端程序的请求,又要等待数据库和其它设备的响应,这将严重影响性能。程序不应该将时间浪费在响应上,而要在等待的同时执行其它任务!

     现在我们开始进入异步编程。在异步程序中,代码不需要按照编写时的顺序执行。这时我们需要用到 C# 5.0 引入的 async/await 来构建异步方法。

 

     我们先看一下不用异步的示例:

class Program{//创建计时器private static readonly Stopwatch Watch = new Stopwatch();private static void Main(string[] args){//启动计时器Watch.Start();const string url1 = "http://www.cnblogs.com/";const string url2 = "http://www.cnblogs.com/liqingwen/";//两次调用 CountCharacters 方法(下载某网站内容,并统计字符的个数)var result1 = CountCharacters(1, url1);var result2 = CountCharacters(2, url2);//三次调用 ExtraOperation 方法(主要是通过拼接字符串达到耗时操作)for (var i = 0; i < 3; i++){ExtraOperation(i + 1);}//控制台输出Console.WriteLine($"{url1} 的字符个数:{result1}");Console.WriteLine($"{url2} 的字符个数:{result2}");Console.Read();}/// <summary>/// 统计字符个数/// </summary>/// <param name="id"></param>/// <param name="address"></param>/// <returns></returns>private static int CountCharacters(int id, string address){var wc = new WebClient();Console.WriteLine($"开始调用 id = {id}:{Watch.ElapsedMilliseconds} ms");var result = wc.DownloadString(address);Console.WriteLine($"调用完成 id = {id}:{Watch.ElapsedMilliseconds} ms");return result.Length;}/// <summary>/// 额外操作/// </summary>/// <param name="id"></param>private static void ExtraOperation(int id){//这里是通过拼接字符串进行一些相对耗时的操作var s = "";for (var i = 0; i < 6000; i++){s += i;}Console.WriteLine($"id = {id} 的 ExtraOperation 方法完成:{Watch.ElapsedMilliseconds} ms");}}

 

                                            

                                                         图1-1 运行的效果图,以毫秒(ms)为单位

 

  【备注】一般来说,直接拼接字符串是一种比较耗性能的手段,如果对字符串拼接有性能要求的话应该使用 StringBuilder。

  【注意】每次运行的结果可能不同。不管哪次调试,绝大部分时间都浪费前两次调用(CountCharacters 方法),即在等待网站的响应上。

 

                                                              图1-2 根据执行结果所画的时间轴

 

     有人曾幻想着这样提高性能的方法:在调用 A 方法时,不等它执行完,直接执行 B 方法,然后等 A 方法执行完成再处理。

     C# 的 async/await 就可以允许我们这么弄。

class Program{//创建计时器private static readonly Stopwatch Watch = new Stopwatch();private static void Main(string[] args){//启动计时器Watch.Start();const string url1 = "http://www.cnblogs.com/";const string url2 = "http://www.cnblogs.com/liqingwen/";//两次调用 CountCharactersAsync 方法(异步下载某网站内容,并统计字符的个数)Task<int> t1 = CountCharactersAsync(1, url1);Task<int> t2 = CountCharactersAsync(2, url2);//三次调用 ExtraOperation 方法(主要是通过拼接字符串达到耗时操作)for (var i = 0; i < 3; i++){ExtraOperation(i + 1);}//控制台输出Console.WriteLine($"{url1} 的字符个数:{t1.Result}");Console.WriteLine($"{url2} 的字符个数:{t2.Result}");Console.Read();}/// <summary>/// 统计字符个数/// </summary>/// <param name="id"></param>/// <param name="address"></param>/// <returns></returns>private static async Task<int> CountCharactersAsync(int id, string address){var wc = new WebClient();Console.WriteLine($"开始调用 id = {id}:{Watch.ElapsedMilliseconds} ms");var result = await wc.DownloadStringTaskAsync(address);Console.WriteLine($"调用完成 id = {id}:{Watch.ElapsedMilliseconds} ms");return result.Length;}/// <summary>/// 额外操作/// </summary>/// <param name="id"></param>private static void ExtraOperation(int id){//这里是通过拼接字符串进行一些相对耗时的操作var s = "";for (var i = 0; i < 6000; i++){s += i;}Console.WriteLine($"id = {id} 的 ExtraOperation 方法完成:{Watch.ElapsedMilliseconds} ms");}}//这是修改后的代码


                                            

                                                                            图1-3 修改后的执行结果图

                                                       图1-4 根据加入异步后的执行结果画的时间轴。

 

  我们观察时间轴发现,新版代码比旧版快了不少(由于网络波动的原因,很可能会出现耗时比之前长的情况)。这是由于 ExtraOperation 方法的数次调用是在 CountCharactersAsync 方法调用时等待响应的过程中进行的。所有的工作都是在主线程中完成的,没有创建新的线程。

 

  【改动分析】只改了几个细节的地方,直接展开代码的话可能看不出来,改动如下:

                               

                                                                                              图1-5

                                     

                                                                                          图1-6

 

  ①从 Main 方法执行到 CountCharactersAsync(1, url1) 方法时,该方法会立即返回,然后才会调用它内部的方法开始下载内容。该方法返回的是一个 Task<int> 类型的占位符对象,表示计划进行的工作。这个占位符最终会返回 int 类型的值。

  ②这样就可以不必等 CountCharactersAsync(1, url1) 方法执行完成就可以继续进行下一步操作。到执行 CountCharactersAsync(2, url2)  方法时,跟 ① 一样返回 Task<int> 对象。

  ③然后,Main 方法继续执行三次 ExtraOperation 方法,同时两次 CountCharactersAsync 方法依然在持续工作 。

  ④t1.Result 和 t2.Result 是指从 CountCharactersAsync 方法调用的 Task<int> 对象取结果,如果还没有结果的话,将阻塞,直有结果返回为止。

 

二、async/await 结构

     先解析一下专业名词:

     同步方法:一个程序调用某个方法,等到其执行完成之后才进行下一步操作。这也是默认的形式。

     异步方法:一个程序调用某个方法,在处理完成之前就返回该方法。通过 async/await 我们就可以实现这种类型的方法。

 

     async/await 结构可分成三部分:

     (1)调用方法:该方法调用异步方法,然后在异步方法执行其任务的时候继续执行;

     (2)异步方法:该方法异步执行工作,然后立刻返回到调用方法;

     (3)await 表达式:用于异步方法内部,指出需要异步执行的任务。一个异步方法可以包含多个 await 表达式(不存在 await 表达式的话 IDE 会发出警告)。

 

  现在我们来分析一下示例。

                           

                                                                                            图2-1

 

 三、What’s 异步方法

     异步方法:在执行完成前立即返回调用方法,在调用方法继续执行的过程中完成任务。

     语法分析:

     (1)关键字:方法头使用 async 修饰。

     (2)要求:包含 N(N>0) 个 await 表达式(不存在 await 表达式的话 IDE 会发出警告),表示需要异步执行的任务。

     (3)返回类型:只能返回 3 种类型(void、Task 和 Task<T>)。Task 和 Task<T> 标识返回的对象会在将来完成工作,表示调用方法和异步方法可以继续执行。

     (4)参数:数量不限,但不能使用 out 和 ref 关键字。

     (5)命名约定:方法后缀名应以 Async 结尾。

     (6)其它:匿名方法和 Lambda 表达式也可以作为异步对象;async 是一个上下文关键字;关键字 async 必须在返回类型前。

                                   

                                                                           图3-1 异步方法的简单结构图

 

小结

  1.解析了进程和线程的概念

  2.异步的简单用法

  3.async/await 结构体

  4.异步方法语法结构


http://chatgpt.dhexx.cn/article/t5h1Sh9D.shtml

相关文章

async/await详解

&#x1f431;个人主页&#xff1a;不叫猫先生 &#x1f64b;‍♂️作者简介&#xff1a;专注于前端领域各种技术&#xff0c;热衷分享&#xff0c;期待你的关注。 &#x1f4ab;系列专栏&#xff1a;vue3从入门到精通 &#x1f4dd;个人签名&#xff1a;不破不立 &#x1f36c…

C# 中的Async 和 Await 的用法详解

1 众所周知C#提供Async和Await关键字来实现异步编程。在本文中&#xff0c;我们将共同探讨并介绍什么是Async 和 Await&#xff0c;以及如何在C#中使用Async 和 Await。 写在前面 自从C# 5.0时代引入async和await关键字后&#xff0c;异步编程就变得流行起来。尤其在现在的.NET…

async、await

词法定义 async async 是“异步”的简写&#xff0c; async 用于申明一个异步的 function await await 可以认为是 async wait 的简写&#xff0c;await 用于等待一个异步方法执行完成。 借检https://www.cnblogs.com/airen123/p/9543490.html 特点&#xff1a; asayc的用…

C# 彻底搞懂async/await

关键&#xff1a; 异步方法&#xff1a;在执行完成前立即返回调用方法&#xff0c;在调用方法继续执行的过程中完成任务。 async/await 结构可分成三部分&#xff1a; &#xff08;1&#xff09;调用方法&#xff1a;该方法调用异步方法&#xff0c;然后在异步方法执行其任务的…

async、await总结

一、async用法 async作为一个关键字放到函数前面&#xff0c;用于表示函数是一个异步函数。异步函数也就意味着该函数的执行不会阻塞后面代码的执行。 异步函数语法很简单&#xff0c;就是在函数前面加上async 关键字&#xff0c;来表示它是异步的。 那怎么调用呢&#xff1f;a…

async和await

async和await 概念 先从字面意思来理解。async 是“异步”的简写&#xff0c;而 await 可以认为是 async wait 的简写。所以应该很好理解 async 用于申明一个 function 是异步的&#xff0c;而 await 用于等待一个异步方法执行完成。 另外还有一个很有意思的语法规定&#xff0…

被说一知半解,await不懂你别乱用

文章目录 1.前言2.await是否阻塞主线程&#xff1f;3.await是否同步&#xff1f;4.总结 1.前言 这篇文章我是不想写的&#xff0c;因为之前已经写过了&#xff08;异步回调Async Await与Promise区别&#xff09;&#xff0c;此时此刻的心情是非常的不满&#xff0c;为什么呢&a…

async、await详解

一、async/await是什么&#xff1f; 简而言之&#xff0c; async用于申明一个function是异步的&#xff1b; 而await则可以认为是 async await的简写形式&#xff0c;是等待一个异步方法执行完成的。 二、async和await的基础使用 async/awiat的使用规则: async 表示这是一个a…

理解异步函数async和await的用法

定义 1. async 是异步的意思&#xff0c;await则可以理解为 async wait。所以可以理解async就是用来声明一个异步方法&#xff0c;而 await是用来等待异步方法执行 2. async作为一个关键字放在函数前面&#xff0c;表示该函数是一个异步函数&#xff0c;异步函数意味着该函数的…

(一)概述:NGS测序在病原微生物检测中的应用

❝ NGS 技术在临床上的应用逐步趋于成熟&#xff0c;从早期的肿瘤基因检测&#xff0c;到如今大热的微生物病原核酸检测&#xff0c;NGS 技术以其快速、准确和高分辨率的特点&#xff0c;发挥着无可替代的作用。 ❞ 微生物在地球上无处不在&#xff0c;从陆地到海洋&#xff0c…

肿瘤NGS测序公司生信工程师水平划分

生信工程师能力划分 1、大头兵&#xff08;1&#xff09;初级生信工程师工作职责任职要求 &#xff08;2&#xff09;中级生信工程师工作职责任职要求 &#xff08;3&#xff09;高级生信工程师工作职责任职要求 2、主管/经理职位描述任职要求 3、总监参考资料&#xff1a; 去年…

NGS数据分析实践:04. 准备测序数据

NGS数据分析实践&#xff1a;04. 准备测序数据 文接上篇&#xff1a;NGS数据分析实践&#xff1a;03. 涉及的常用数据格式[5] - vcf格式 本次NGS数据&#xff0c;采用多重PCR靶向扩增子测序技术&#xff0c;在Illunima Hiseq X Ten测序平台上进行双末端&#xff08;paired-end…

NGS_panel的CAP认证学习笔记

对于基因的定义总体可以划分为两类 GAD: Gene associated with Mendelian disorder; GADs include genes that meet criteria for definitive, strong, or moderate evidence for association with disease as described by ClinGenGUS: Gene of uncertain significance; GUSs …

45万例患者基因检测显示:NGS很难检测出七分之一的致病变异

基于下一代测序&#xff08;NGS&#xff09;的临床基因测试越来越多地用于辅助诊断&#xff0c;针对该技术的临床应用有具体的指南&#xff0c;除了检测相对可靠的单核苷酸变异&#xff08;SNVs&#xff09;和微小插入缺失&#xff08;indel&#xff09;&#xff0c;NGS也已经被…

R实战 | NGS数据时间序列分析(maSigPro)

masigpro 跟着Cell学作图 | 6.时间序列分析(Mfuzz包) 一个答疑教程。 maSigPro 流程 示例数据 #BiocManager::install(maSigPro) library(maSigPro) # 载入示例数据 data(data.abiotic) data.abiotic[1:5,1:5] data(edesign.abiotic) head(edesign.abiotic) > data.abiotic…

NGS数据分析实践:03. 涉及的常用数据格式[5] - vcf格式

NGS数据分析实践&#xff1a;03. 涉及的常用数据格式[5] - vcf格式 6. vcf格式6.1 vcf格式整体描述6.2 第8列INFO详解6.3 第9列FORMAT详解6.4 vcf文件简单解读 系列文章&#xff1a; 二代测序方法&#xff1a;DNA测序之靶向重测序 NGS数据分析实践&#xff1a;00. 变异识别的基…

生信小白学习日记Day4Day5——NGS基础 NGS分析注释(BWA软件)

2019年5月30日&#xff0c;晚上&#xff0c;心情变好&#xff0c;好几天没更新了&#xff0c;看到男朋友在学一款软件&#xff0c;我也近朱者赤&#xff0c;来继续注释Day2-2中NGS分析流程中的一个重要软件——BWA NGS基础 NGS分析注释 BWA 对应于NGS分析流程的这两步&…

NGS数据分析实践:00. 变异识别的基本流程

NGS数据分析实践&#xff1a;00. 变异识别的基本流程 变异识别过程可以分成3大块&#xff1a;1. 原始数据质控&#xff1b;2. 数据预处理&#xff1b;3. 变异识别。大致可以细分为6个部分&#xff1a;(1) 原始测序数据的质控&#xff1b;(2) read比对&#xff0c;排序和标记PCR…

如何用软件模拟NGS数据

如何用软件模拟NGS数据 为了评价一个工具的性能&#xff0c;通常我们都需要先模拟一批数据。这样相当于有了参考答案&#xff0c;才能检查工具的实际表现情况。因此对于我们而言&#xff0c;面对一个新的功能&#xff0c;可以先用模拟的数据测试下不同工具的优缺点。有如下几个…

生信小白学习日记Day2——NGS基础 illumina高通量测序原理

2019年5月26日&#xff0c;周日&#xff0c;小雨 说明&#xff1a;阅读生信宝典和查阅文章的总结&#xff0c;原文请关注公众号生信宝典&#xff0c;参考的博文都附有链接&#xff0c;仅供参考。 生信宝典 NGS基础——高通量测序原理 本文介绍了测序文库构建原理、链特异性文…