您当前的位置:首页 > 电脑百科 > 程序开发 > 语言 > C/C++/C#

C# await 高级用法

时间:2021-05-25 10:36:39  来源:今日头条  作者:IT狂人日记

本文告诉大家 await 的高级用法,包括底层原理。

昨天看到太子写了一段代码,我开始觉得他修改了编译器,要不然下面的代码怎么可以编译通过

await "林德熙逗比";

需要知道,基本可以添加 await 都是可以等待的类型,如 Task 。如果一个类需要可以被等待,那么这个类必须满足以下条件

  • 类里有一个 GetAwaiter 函数
  • GetAwaiter 有返回值,返回值需要继承 INotifyCompletion 并且有 bool IsCompleted { get; },GetResult(),void OnCompleted(Action continuation) 定义

参见:如何实现一个可以用 await 异步等待的 Awaiter - walterlv

但是上面的代码使用的是一个字符串,什么时候可以修改继承字符串?

先让我来说下 await 原理,因为知道了原理,上面的代码实现很简单。看完了本文,你就会知道如何让几乎所有类型包括 int 、string 、自定义的类都支持 await 。

如果真的不想看原理,那么请直接调到文章的最后,看到最后很快就知道如何做。

原理

在 .net 4.5 之后,框架默认提供 async 和 await 的语法糖,这时千万不要认为进入 await 就会进入一个新的线程,实际上不一定会进入一个新的线程才会调用 await 。

那么 await 的语法糖写的是什么?实际上就是以前的 Begin xx 和 End xx 的语法糖。

古时候的写法:

foo.Beginxx();

foo.Endxx(传入委托);

这样大家就无法在一个流程写完,需要分为两个东西,而在 Continus with 下,就需要传入委托。如果委托里又使用了异步,那么又需要传入委托

       task.ContinueWith(_ =>
            {
                Task t1 = new Task(() => { });
                t1.ContinueWith((t2) =>
                {
                    //可以看到如果进入很多的委托
                });
            });

所以这时就使用了 await ,可以让大家按照顺序写。

await task;
Task t1 = new Task(() => { });
await t1;
//可以看到这时不需要进入委托

实际上 await 是在编译时支持的,请看进阶篇:以IL为剑,直指async/await - 布鲁克石 - 博客园

而且千万不要认为 await 一定会进入一个新的线程,实际上他只是把需要写在多处的代码,可以按照流写下载,和写同步代码一样。如果感兴趣 await 不一定会进入一个新的线程请看 There Is No Thread

使用

因为 await 需要找到一个 GetAwaiter 函数,这个函数即使是扩展方法也可以,所以其实上面的代码是这样写的


    public static class KvpbamjhKsvm
    {
        public static HeabdsdnbKevx GetAwaiter(this string str)
        {
            return new HeabdsdnbKevx();
        }
    }

    public class HeabdsdnbKevx : INotifyCompletion
    {
        public bool IsCompleted { get; }

        public void GetResult()
        {
        }

        /// <inheritdoc />
        public void OnCompleted(Action continuation)
        {
        }
    }

HeabdsdnbKevx 就是一个可以等待的类型

现在就可以写出下面的代码

        private static void Main(string[] args)
        {
            DdngSiwchjux();
        }

        private static async void DdngSiwchjux()
        {
            await "林德熙逗比";
        }

当然,上面的这个代码可以运行,不过不会返回什么。下面让我加上一句代码。

        private static void Main(string[] args)
        {
            DdngSiwchjux();
        }

        private static async void DdngSiwchjux()
        {
            await "林德熙逗比";
            Console.WriteLine("csdn");
        }

这时可以看到,Console.WriteLine("csdn");不会运行,因为这时如果在 OnCompleted 函数打断点就可以看到,执行await "林德熙逗比"之后就进入OnCompleted 函数。从上面的原理可以知道,这个函数传入的参数就是两个await或 await和函数结束之间的代码。如果需要让Console.WriteLine("csdn");运行,那么只需要在OnCompleted运行参数

C# await 高级用法

 

   public void OnCompleted(Action continuation)
        {
            continuation();
        }
C# await 高级用法

 

但是作为一个挖坑专业的大神,怎么可以就扩展 string ,下面我把 int 进行扩展

    public static class KvpbamjhKsvm
    {
        public static HeabdsdnbKevx GetAwaiter(this int dxpbnzSce)
        {
            return new HeabdsdnbKevx();
        }
    }

随意写一个值,然后进行等待

C# await 高级用法

 

现在我准备在 object 加一个扩展方法,所有类型都可以等待,然后把这个扩展方法的 namespace 写为 System ,这样大家就不知道这个是我写的,过了一年我就告诉大家这是 C# 的特性,所有的类都可以等待。但是这个特性需要开光才可以使用,你们直接建的项目没有开光所以没法使用这个特性。

等待和不等待的区别

虽然很多时候从原理上看,等待和不等待只是调用时机的问题。但是依旧遇到一些小伙伴一直以为全部的异步方法都需要await,看到我写了没有直接await的代码觉得很诡异,所以我在这里做个实验给大家看。

下面的代码是最常见的代码,在 async Task 的方法使用 await ,这样就会等待这个方法完成,代码就和同步代码一样。

 await GagarLerecel();
private static async Task GagarLerecel()

例如我这样写

            await GagarLerecel();

        private static async Task GagarLerecel()
        {
            Write("GagarLerecel 开始");
            await Task.Delay(100);
            Write("GagarLerecel 完成");
        }

输出就是按照顺序输出

GagarLerecel 开始
GagarLerecel 完成

如果我修改一下代码,创建一个新的函数 CoujafuDarso 里面的代码和上面函数相同

        private static async Task CoujafuDarso()
        {
            Write("CoujafuDarso开始");
            await Task.Delay(100);
            Write("CoujafuDarso结束");
        }

但是不在调用 CoujafuDarso 使用 await ,而是使用一个变量

            var aa = CoujafuDarso();
            Write("其他代码");
            await aa;

就是这样的代码,我的小伙伴说,这样写不清真,实际上这样写也是清真的代码。在调用 CoujafuDarso 会在代码到第一个 await 函数就返回,于是先执行了CoujafuDarso开始,然后函数返回,执行Write("其他代码"),在最后await aa才等待函数把所有代码执行完成。所以可以看到下面输出

CoujafuDarso开始
其他代码
CoujafuDarso结束

但是不加 await 的呢?也就是函数一直都没有等待,我再写一个函数BotujawWorpay

        private static async Task BotujawWorpay()
        {
            Write("BotujawWorpay开始");
            await Task.Delay(100);
            Write("BotujawWorpay结束");
        }

调用的时候没有等待

            BotujawWorpay();
            Write("CesearJemmeme");

这时会在输出CesearJemmeme之后,某个时间继续执行函数

BotujawWorpay开始
CesearJemmeme
BotujawWorpay结束

这样和使用 void 函数有什么区别?

在执行的函数遇到第一个 await 就会返回,这样就可以继续执行函数下面的代码

C# await 高级用法

 

输出下面代码

德熙逗比代码
BarpooseewhowGelpousacall 代码1 线程1
德熙逗比状态开始
BarpooseewhowGelpousacall 代码2 线程5
BarpooseewhowGelpousacall 代码3 线程4
BarpooseewhowGelpousacall 完成 线程5

多线程

不是所有的 await 都会开多线程,如下面的代码

       static void Main(string[] args)
        {
            Write("开始");
            Write("线程" + Thread.CurrentThread.ManagedThreadId);

            CeaXisci();
            Task.Run(async () =>
            {
                await Task.Delay(1000);
                MouvaypuNasjo();
            });
            while (true)
            {
                Console.Read();
            }
        }

        private static async Task BarpooseewhowGelpousacall()
        {
            Write("BarpooseewhowGelpousacall 代码1 线程" + Thread.CurrentThread.ManagedThreadId);
            await Task.Delay(10);
            Write("BarpooseewhowGelpousacall 代码2 线程" + Thread.CurrentThread.ManagedThreadId);
            await Task.Delay(10);
            Write("BarpooseewhowGelpousacall 完成 线程" + Thread.CurrentThread.ManagedThreadId);
        }

也就是在没有Task.Delay分开的代码,只要使用了 await 那么就可以在同个线程运行,请看输出。在最后的BarpooseewhowGelpousacall 完成和这个函数后面的代码都在同一个线程运行,而上面的代码,可能是在同个线程,也可能在不同的线程

开始
线程1
CeaXisci 开始 线程1
BarpooseewhowGelpousacall 代码1 线程1
BarpooseewhowGelpousacall 代码2 线程5
BarpooseewhowGelpousacall 完成 线程4
CeaXisci 开始 完成4

本文会经常更新,请阅读原文:
https://blog.lindexi.com/post/C-await-%E9%AB%98%E7%BA%A7%E7%94%A8%E6%B3%95.html ,以避免陈旧错误知识的误导,同时有更好的阅读体验。



Tags:C#   点击:()  评论:()
声明:本站部分内容及图片来自互联网,转载是出于传递更多信息之目的,内容观点仅代表作者本人,如有任何标注错误或版权侵犯请与我们联系(Email:2595517585@qq.com),我们将及时更正、删除,谢谢。
▌相关推荐
本文告诉大家 await 的高级用法,包括底层原理。昨天看到太子写了一段代码,我开始觉得他修改了编译器,要不然下面的代码怎么可以编译通过await "林德熙逗比";需要知道,基本可以添...【详细内容】
2021-05-25  Tags: C#  点击:(218)  评论:(0)  加入收藏
Below are our C# coding standards, naming conventions, and best practices. Use these in your own projects and/or adjust these to your own needs. douse PascalCas...【详细内容】
2020-08-24  Tags: C#  点击:(116)  评论:(0)  加入收藏
介绍本文主要为C#开发人员了解Java提供一些基础。Java中缺少C#的功能 C#包含更多原始类型和捕获算术异常的功能。 包括大量的Java注释便利,其中许多,例如运算符重载和用户定义的...【详细内容】
2020-03-10  Tags: C#  点击:(62)  评论:(0)  加入收藏
▌简易百科推荐
一、简介很多时候我们都需要用到一些验证的方法,有时候需要用正则表达式校验数据时,往往需要到网上找很久,结果找到的还不是很符合自己想要的。所以我把自己整理的校验帮助类分...【详细内容】
2021-12-27  中年农码工    Tags:C#   点击:(1)  评论:(0)  加入收藏
引言在学习C语言或者其他编程语言的时候,我们编写的一个程序代码,基本都是在屏幕上打印出 hello world ,开始步入编程世(深)界(坑)的。C 语言版本的 hello world 代码:#include <std...【详细内容】
2021-12-21  一起学嵌入式    Tags:C 语言   点击:(10)  评论:(0)  加入收藏
读取SQLite数据库,就是读取一个路径\\192.168.100.**\position\db.sqlite下的文件<startup useLegacyV2RuntimeActivationPolicy="true"> <supportedRuntime version="v4.0"/...【详细内容】
2021-12-16  今朝我的奋斗    Tags:c#   点击:(21)  评论:(0)  加入收藏
什么是shellshell是c语言编写的程序,它在用户和操作系统之间架起了一座桥梁,用户可以通过这个桥梁访问操作系统内核服务。 它既是一种命令语言,同时也是一种程序设计语言,你可以...【详细内容】
2021-12-16  梦回故里归来    Tags:shell脚本   点击:(16)  评论:(0)  加入收藏
一、编程语言1.根据熟悉的语言,谈谈两种语言的区别?主要浅谈下C/C++和PHP语言的区别:1)PHP弱类型语言,一种脚本语言,对数据的类型不要求过多,较多的应用于Web应用开发,现在好多互...【详细内容】
2021-12-15  linux上的码农    Tags:c/c++   点击:(17)  评论:(0)  加入收藏
1.字符串数组+初始化char s1[]="array"; //字符数组char s2[6]="array"; //数组长度=字符串长度+1,因为字符串末尾会自动添&lsquo;\0&lsquo;printf("%s,%c\n",s1,s2[2]);...【详细内容】
2021-12-08  灯-灯灯    Tags:C语言   点击:(46)  评论:(0)  加入收藏
函数调用约定(Calling Convention),是一个重要的基础概念,用来规定调用者和被调用者是如何传递参数的,既调用者如何将参数按照什么样的规范传递给被调用者。在参数传递中,有两个很...【详细内容】
2021-11-30  小智雅汇    Tags:函数   点击:(19)  评论:(0)  加入收藏
一、问题提出问题:把m个苹果放入n个盘子中,允许有的盘子为空,共有多少种方法?注:5,1,1和1 5 1属同一种方法m,n均小于10二、算法分析设f(m,n) 为m个苹果,n个盘子的放法数目,则先对...【详细内容】
2021-11-17  C语言编程    Tags:C语言   点击:(46)  评论:(0)  加入收藏
一、为什么需要使用内存池在C/C++中我们通常使用malloc,free或new,delete来动态分配内存。一方面,因为这些函数涉及到了系统调用,所以频繁的调用必然会导致程序性能的损耗;另一...【详细内容】
2021-11-17  深度Linux    Tags:C++   点击:(37)  评论:(0)  加入收藏
OpenCV(Open Source Computer Vision Library)是一个(开源免费)发行的跨平台计算机视觉库,可以运行在Linux、Windows、Android、ios等操作系统上,它轻量级而且高效---由一系列...【详细内容】
2021-11-11  zls315    Tags:C#   点击:(50)  评论:(0)  加入收藏
最新更新
栏目热门
栏目头条