设为首页收藏本站
网站公告 | 这是第一条公告
     

 找回密码
 立即注册
缓存时间15 现在时间15 缓存数据 一个人挺好的

一个人挺好的 -- 一个

查看: 645|回复: 0

C#中async await异步关键字用法和异步的底层原理全解析

[复制链接]

  离线 

TA的专栏

等级头衔

等級:晓枫资讯-列兵

在线时间
0 小时

积分成就
威望
0
贡献
31
主题
25
精华
0
金钱
93
积分
60
注册时间
2023-9-29
最后登录
2025-6-1

发表于 2025-6-1 05:27:57 | 显示全部楼层 |阅读模式
C#异步编程


一、异步编程基础

异步编程是啥玩意儿

  • 就是让程序在干等着某些耗时操作(比如等网络响应、读写文件啥的)的时候,能把线程腾出来干别的活儿,这样程序就能更灵敏、更高效啦。
  • 跟同步编程不一样,同步编程就是老老实实等着操作完成才继续往下走,线程就一直被占着,多浪费啊。
异步编程的好处

  • 响应快:比如在做UI界面的时候,用了异步编程,界面就不会卡啦,用户体验贼棒。
  • 省资源:不用让线程一直干等着,资源利用率就上去了。
  • 能扛更多活儿:面对一大堆并发操作的时候,异步编程能轻松搞定,扩展性杠杠滴。

二、异步方法的工作原理

异步方法咋被编译的

  • 你写个
    1. async
    复制代码
    修饰的方法,编译器就把它变成一个状态机啦。
  • 状态机会根据
    1. await
    复制代码
    表达式把方法拆成好多个状态,就跟玩拼图一样。
状态机是咋干活的

  • 状态机就是编译器生成的一个类,它得记着异步方法执行到哪儿了。
  • 核心就是
    1. MoveNext
    复制代码
    方法,它就像导演一样,指挥着异步操作一步步往下走。
  • 每碰到一个
    1. await
    复制代码
    ,就切换一下状态。
  1. await
复制代码
底层是咋实现的

    1. await
    复制代码
    就整出个等待器(awaiter),专门等着异步操作完成。
  • 要是操作还没完,
    1. await
    复制代码
    就记下当前状态,等操作完了再继续往下走。

三、代码示例

  1. HttpClient
复制代码
干异步网络请求

  • 弄个
    1. HttpClient
    复制代码
    对象,用来发HTTP请求。
    1. GetStringAsync
    复制代码
    方法,就能异步拿到指定URL的网页内容啦。
  • 把拿到的内容打印出来,瞧一瞧成果。
  1. using System;
  2. using System.Net.Http;
  3. using System.Threading.Tasks;
  4. namespace asyncawait原理1
  5. {
  6.     class Program
  7.     {
  8.         static async Task Main(string[] args)
  9.         {
  10.             using (HttpClient httpClient = new HttpClient())
  11.             {
  12.                 string html = await httpClient.GetStringAsync("https://www.baidu.com");
  13.                 Console.WriteLine(html);
  14.             }
  15.         }
  16.     }
  17. }
复制代码
异步读写文件

    1. File.WriteAllTextAsync
    复制代码
    方法,能把文本异步写到指定路径的文件里。
    1. File.ReadAllTextAsync
    复制代码
    方法,就能把文件内容异步读出来。
  • 把读到的内容打印出来,看看对不对。
  1. using System;
  2. using System.IO;
  3. using System.Threading.Tasks;
  4. namespace asyncawait原理1
  5. {
  6.     class Program
  7.     {
  8.         static async Task Main(string[] args)
  9.         {
  10.             string txt = "hello world";
  11.             string filename = @"E:\temp\1.txt";
  12.             await File.WriteAllTextAsync(filename, txt);
  13.             Console.WriteLine("写入成功");
  14.             string s = await File.ReadAllTextAsync(filename);
  15.             Console.WriteLine("文件内容:" + s);
  16.         }
  17.     }
  18. }
复制代码
四、编译后的底层实现

用ILSpy反编译DLL文件

  • ILSpy就是个反编译工具,能把DLL文件变回C#代码,方便咱们研究。
  • 把DLL文件加载到ILSpy里,就能看到编译后的代码啦。
  1. [CompilerGenerated]
  2. private sealed class <>c__DisplayClass0_0 : IAsyncStateMachine
  3. {
  4.     public int <>1__state;
  5.     public AsyncTaskMethodBuilder <>t__builder;
  6.     public string[] args;
  7.     private string <>s__1;
  8.     private string <>s__3;
  9.     private string <>s__6;
  10.     private HttpClient <httpClient>__4;
  11.     private string <html>__5;
  12.     private string <txt>__2;
  13.     private string <filename>__7;
  14.     private void MoveNext()
  15.     {
  16.         int num = this.<>1__state;
  17.         try
  18.         {
  19.             TaskAwaiter<string> awaiter;
  20.             TaskAwaiter awaiter2;
  21.             switch (num)
  22.             {
  23.                 default:
  24.                     this.<httpClient>__4 = new HttpClient();
  25.                     goto case 0;
  26.                 case 0:
  27.                     try
  28.                     {
  29.                         awaiter = this.<httpClient>__4.GetStringAsync("https://www.baidu.com").GetAwaiter();
  30.                         if (!awaiter.IsCompleted)
  31.                         {
  32.                             num = this.<>1__state = 0;
  33.                             this.<>t__builder.AwaitUnsafeOnCompleted(ref awaiter, ref this);
  34.                             return;
  35.                         }
  36.                     }
  37.                     catch (Exception exception)
  38.                     {
  39.                         this.<>1__state = -2;
  40.                         this.<>t__builder.SetException(exception);
  41.                         return;
  42.                     }
  43.                     this.<html>__5 = awaiter.GetResult();
  44.                     Console.WriteLine(this.<html>__5);
  45.                     this.<txt>__2 = "hello yz";
  46.                     this.<filename>__7 = @"E:\temp\1.txt";
  47.                     awaiter2 = File.WriteAllTextAsync(this.<filename>__7, this.<txt>__2).GetAwaiter();
  48.                     if (!awaiter2.IsCompleted)
  49.                     {
  50.                         num = this.<>1__state = 1;
  51.                         this.<>t__builder.AwaitUnsafeOnCompleted(ref awaiter2, ref this);
  52.                         return;
  53.                     }
  54.                     break;
  55.                 case 1:
  56.                     awaiter2 = this.<>s__1;
  57.                     this.<>s__1 = null;
  58.                     num = this.<>1__state = -1;
  59.                     break;
  60.             }
  61.             awaiter2.GetResult();
  62.             Console.WriteLine("写入成功");
  63.             this.<>s__3 = null;
  64.             awaiter = File.ReadAllTextAsync(this.<filename>__7).GetAwaiter();
  65.             if (!awaiter.IsCompleted)
  66.             {
  67.                 num = this.<>1__state = 2;
  68.                 this.<>t__builder.AwaitUnsafeOnCompleted(ref awaiter, ref this);
  69.                 return;
  70.             }
  71.             this.<>s__6 = awaiter.GetResult();
  72.             Console.WriteLine("文件内容:" + this.<>s__6);
  73.             this.<>s__6 = null;
  74.             this.<>t__builder.SetResult();
  75.         }
  76.         catch (Exception exception)
  77.         {
  78.             this.<>1__state = -2;
  79.             this.<>t__builder.SetException(exception);
  80.             return;
  81.         }
  82.         this.<>1__state = -1;
  83.     }
  84.     void IAsyncStateMachine.MoveNext()
  85.     {
  86.         // This method is implemented by the compiler-generated code.
  87.     }
  88.     [DebuggerHidden]
  89.     private void SetStateMachine(IAsyncStateMachine stateMachine)
  90.     {
  91.         this.<>t__builder.SetStateMachine(stateMachine);
  92.     }
  93.     void IAsyncStateMachine.SetStateMachine(IAsyncStateMachine stateMachine)
  94.     {
  95.         this.SetStateMachine(stateMachine);
  96.     }
  97. }
复制代码
看看编译后的状态机代码

  • 分析状态机类的结构,看看都有啥变量、
    1. MoveNext
    复制代码
    方法长啥样。
  • 瞧瞧
    1. awaiter
    复制代码
    咋用的,状态咋切换的。
理解
  1. MoveNext
复制代码
方法是干啥的

    1. MoveNext
    复制代码
    就是状态机的发动机,它决定了异步方法咋执行。
  • 在这个方法里,会根据当前状态执行对应的代码,碰到
    1. await
    复制代码
    就暂停,安排好后续咋继续。

五、总结

异步方法编译过程回顾

  • 再唠唠
    1. async
    复制代码
    方法咋被编译成状态机的,状态机又咋根据
    1. await
    复制代码
    拆分方法、驱动异步操作的。
  1. await
复制代码
到底在干啥

  • 说白了,
    1. await
    复制代码
    根本不是真的“等待”,而是靠状态机和等待器来实现的异步协作。
  • 强调一下异步编程的好处,比如响应快、省资源、能扛更多活儿,还有啥场景适合用它。
到此这篇关于C#中async await异步关键字用法和异步的底层原理的文章就介绍到这了,更多相关C# async await异步关键字内容请搜索晓枫资讯以前的文章或继续浏览下面的相关文章希望大家以后多多支持晓枫资讯!

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!
晓枫资讯-科技资讯社区-免责声明
免责声明:以上内容为本网站转自其它媒体,相关信息仅为传递更多信息之目的,不代表本网观点,亦不代表本网站赞同其观点或证实其内容的真实性。
      1、注册用户在本社区发表、转载的任何作品仅代表其个人观点,不代表本社区认同其观点。
      2、管理员及版主有权在不事先通知或不经作者准许的情况下删除其在本社区所发表的文章。
      3、本社区的文章部分内容可能来源于网络,仅供大家学习与参考,如有侵权,举报反馈:点击这里给我发消息进行删除处理。
      4、本社区一切资源不代表本站立场,并不代表本站赞同其观点和对其真实性负责。
      5、以上声明内容的最终解释权归《晓枫资讯-科技资讯社区》所有。
http://bbs.yzwlo.com 晓枫资讯--游戏IT新闻资讯~~~
严禁发布广告,淫秽、色情、赌博、暴力、凶杀、恐怖、间谍及其他违反国家法律法规的内容。!晓枫资讯-社区
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|晓枫资讯--科技资讯社区 本站已运行

CopyRight © 2022-2025 晓枫资讯--科技资讯社区 ( BBS.yzwlo.com ) . All Rights Reserved .

晓枫资讯--科技资讯社区

本站内容由用户自主分享和转载自互联网,转载目的在于传递更多信息,并不代表本网赞同其观点和对其真实性负责。

如有侵权、违反国家法律政策行为,请联系我们,我们会第一时间及时清除和处理! 举报反馈邮箱:点击这里给我发消息

Powered by Discuz! X3.5

快速回复 返回顶部 返回列表