广告
返回顶部
首页 > 资讯 > 精选 >C#怎么使用async和await实现异步编程
  • 472
分享到

C#怎么使用async和await实现异步编程

2023-07-02 17:07:26 472人浏览 独家记忆
摘要

这篇文章主要介绍“C#怎么使用async和await实现异步编程”,在日常操作中,相信很多人在C#怎么使用async和await实现异步编程问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”C#怎么使用async

这篇文章主要介绍“C#怎么使用async和await实现异步编程”,在日常操作中,相信很多人在C#怎么使用async和await实现异步编程问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”C#怎么使用async和await实现异步编程”的疑惑有所帮助!接下来,请跟着小编一起来学习吧!

1.普通的程序怎么写?

class Program{    static void Main(string[] args)    {        MyDownLoadString ds = new MyDownLoadString();        ds.DoRun();        Console.ReadKey();    }     class MyDownLoadString    {        Stopwatch sw = new Stopwatch();        public void DoRun()        {            const int LargeNumber = 6000000;            sw.Start();             int t1 = CountCharacters(1, "Http://www.microsoft.com");            int t2 = CountCharacters(2, "http://www.illustratedcsharp.com");             CountToALargeNumber(1, LargeNumber);            CountToALargeNumber(2, LargeNumber);            CountToALargeNumber(3, LargeNumber);            CountToALargeNumber(4, LargeNumber);                       Console.WriteLine("Chars in Call1:{0}",t1);            Console.WriteLine("Chars in Call1:{0}",t2);        }         private int CountCharacters(int id, string uriString)        {            WEBClient wc1 = new WebClient();            Console.WriteLine("Call {0} start: {1:N0}ms ", id, sw.Elapsed.TotalMilliseconds);            string result = wc1.DownloadString(new Uri(uriString));            Console.WriteLine("Call {0} completed: {1:N0}ms", id, sw.Elapsed.TotalMilliseconds);            return result.Length;        }         private void CountToALargeNumber(int id, int value)        {            for (long i = 0; i < value; i++) ;            Console.WriteLine("End CountToALargeNumber {0} : {1:N0}ms", id, sw.Elapsed.TotalMilliseconds);        }    }}

结果:

Call 1 start: 1ms
Call 1 completed: 903ms
Call 2 start: 903ms
Call 2 completed: 1,355ms
End CountToALargeNumber 1 : 1,375ms
End CountToALargeNumber 2 : 1,399ms
End CountToALargeNumber 3 : 1,417ms
End CountToALargeNumber 4 : 1,435ms
Chars in Call1:161702
Chars in Call1:5164

从运行结果可以看到,同步执行的时间主要花在了两次请求外部地址上,计算长度并不费时,用图来表示就像下面

C#怎么使用async和await实现异步编程

2.使用async和await怎么写?

修改上面代码,如下

class MyDownLoadString{    Stopwatch sw = new Stopwatch();    public void DoRun()    {        const int LargeNumber = 6000000;        sw.Start();         // Task<int> 保存结果对象,后面t1.Result则是获取结果         Task<int> t1 = CountCharactersAsync(1, "http://www.microsoft.com");        Task<int> t2 = CountCharactersAsync(2, "http://www.illustratedcsharp.com");               //无需等待CountCharactersAsync执行完成        CountToALargeNumber(1, LargeNumber);        CountToALargeNumber(2, LargeNumber);        CountToALargeNumber(3, LargeNumber);        CountToALargeNumber(4, LargeNumber);                 //t1.Result获取结果        Console.WriteLine("Chars in Call1:{0}",t1.Result);        Console.WriteLine("Chars in Call1:{0}",t2.Result);    }     private async Task<int> CountCharactersAsync(int id, string uriString)    {        WebClient wc = new WebClient();        Console.WriteLine("Call {0} start: {1:N0}ms ", id, sw.Elapsed.TotalMilliseconds);        string result = await wc.DownloadStringTaskAsync(new Uri(uriString));        Trace.TraceInfORMation("Taceing Async Call {0} @time:{1:N0}ms", id, sw.Elapsed.TotalMilliseconds);        Console.WriteLine("Call {0} completed: {1:N0}ms", id, sw.Elapsed.TotalMilliseconds);        return result.Length;    }     private void CountToALargeNumber(int id, int value)    {        for (long i = 0; i < value; i++) ;        Console.WriteLine("End CountToALargeNumber {0}: {1:N0}ms", id, sw.Elapsed.TotalMilliseconds);    }}

运行结果:

Call 1 start: 2ms
Call 2 start: 253ms
End CountToALargeNumber 1: 288ms
End CountToALargeNumber 2: 359ms
End CountToALargeNumber 3: 560ms
Call 1 completed: 770ms
End CountToALargeNumber 4: 844ms
Call 2 completed: 887ms
Chars in Call1:162262
Chars in Call2:5164

修改如上面的代码之后,我们就可以无需等待两次CountCharactersAsync返回结果,而是直接调用了下面的CountToALargeNumber,在CountCharactersAsync请求返回的时候再获取结果。

C#怎么使用async和await实现异步编程

3.async和await的细节

async和await可以创建和使用异步方法,这个特性的由三个部分组成:

  • ①调用方法(calling method):该方法调用异步方法,然后在异步方法(可能使用同一个线程也可能不在一个线程)执行其任务的时候继续执行

  • ②异步方法(async): 该方法异步执行其工作,然后立即方法到调用方法

  • ③await表达式:用于异步方法内部,指明需要异步执行的惹怒我。一个异步方法可以包含任意多个await表达式,如果一个都不包含编译器会发出警告

举例说明一个async/await方法:

//1.调用方法static void Main(string[] args){    Task<int> t = DoSumAsync(1, 2);    Console.WriteLine("结果:{0}", t.Result);    Console.ReadKey();} //2.异步方法public static async Task<int> DoSumAsync(int a, int b){    //3.await 表达式    int sum = await Task.Run(() => { return a + b; });    return sum;}

4.什么是异步方法?

上面简单举例了什么是异步方法,下面就详细学习一下:

异步方法在完成其工作之前返回到调用方法,并在调用方法继续执行的时候完成其工作。语法上有如下特征:

  • ① 方法使用async作为修饰符

  • ② 方法内部包含一个或者多个await表达式,表示可以异步完成的任务

  • ③ 必须具备以下三种返回类型 void 、Task 、Task<T> ,其中后两种的返回对象标识讲座未来完成的工作,调用方法和异步方法可以继续执行。

  • ④异步方法的参数可以任意类型,但是不能为out和ref参数

  • ⑤约定俗成,一般异步方法都是以 Async作为后缀的。

  • ⑥ 除了方法之外,Lambda表达式和匿名函数也可以作为异步对象。

像代码:

private async Task<int> CountCharactersAsync(int id, string uriString){    WebClient wc = new WebClient();    Console.WriteLine("Call {0} start: {1:N0}ms ", id, sw.Elapsed.TotalMilliseconds);    string result = await wc.DownloadStringTaskAsync(new Uri(uriString));    Trace.TraceInformation("Taceing Async Call {0} @time:{1:N0}ms", id, sw.Elapsed.TotalMilliseconds);    Console.WriteLine("Call {0} completed: {1:N0}ms", id, sw.Elapsed.TotalMilliseconds);    return result.Length;}

详细说明:

①async关键字是一个上下文关键字,也就是说除了做为方法(lambda和匿名函数)的修饰符之外,还可以做标识符。

②返回类型

  • Task类型:如果调用方法不需要从异步方法中返回某个值,但需要检查异步方法的状态,可以返回一个Task,此时就算异步方法中出现了return语句,也不会返回任何东西。

  • Task<T>类型,除了上面Task的功能,还可以通过 Return属性来获取返回的T类型的值。

  • void类型:如果仅仅是执行异步方法,而不需要与它做任何进一步的交互(“调用并忘记”),此时可以用void,和Task一样,就算有return语句,也得不到任何东西。

5.异步方法的控制流

首先要明确“异步方法”的三个部分,如下图所示:

  • ①首先是第一个await之前的部分,这部分应该是少量且无需长时间等待的代码。

  • ②await表达式,表示需要被异步执行的任务,这里有两个await表达式,第二个await和之前的同步部分和第一个await以及之前的部分是一样的。

  • ③后续部分:在await表达式之后出现的方法中的其余代码。

C#怎么使用async和await实现异步编程

执行过程,可以参考下面的图

C#怎么使用async和await实现异步编程

有几个注意的地方:

  • ① await之前的部分是同步执行的

  • ② 当达到awati的时候,会将异步方法的控制返回给调用方法。如果方法返回的类型是Task或者Task<T>,将创建一个Task对象,表示需异步完成的任务和后续,然后将该Task返回到调用方法。 这里的返回值并不是await表达式的返回值,而是异步方法中声明的返回值类型。

  • ③ 异步方法内部需要完成以下工作:

  - 异步执行await表达是的空闲任务

       - 当await表达式执行完成之后,执行后续部分。后续本身也可能是await表达式,处理过程和上一个一致。

  - 后续部分如果遇到 return 或者 方法达到末尾,将做如下的事情:

    l  如果返回的类型是void,控制流就退出了

    l  如果返回的类型是Task,后续部分设置Task对象的属性并退出。

    l  如果返回的类型是Task<T>,不仅要设置Task对象属性,还要设置Task对象的Return属性。

    这个点要注意下:并不是遇到return或者达到方法末尾,就能获取到返回值,它只是退出了。

  • ④ 调用方法继续执行,会从异步方法获取Task对象。当需要其实际值的时候,就引用Task对象中的Result属性。届时,如果异步方法设置了该属性,调用方法获取其值并继续。否则就等待该属性被设置,然后再继续执行。

6. await表达式

await表达式指定了一个异步执行的任务。语法由 await关键字 + 一个空闲对象(称为任务)组成。这个任务可能是一个Task对象,也可以不是,默认情况下由该线程异步执行。

一个空闲对象 指的是一个awaitable类型的实例,awaitable类型是指包含了GetAwaiter方法的类型,方法没有参数,返回一个称为awaiter类型的对象。

C#怎么使用async和await实现异步编程

一个awaiter对象包含了如下成员:

C#怎么使用async和await实现异步编程

一般情况下我们不需要自己构建一个awaiter对象,使用.net 自己的Task就可以了。最简单的方法就是使用Task.Run()来返回一个Task对象。关于Task.Run()有一个非常重要的点,他将在不同的线程上运行你的方法。

6.异常处理和await表达式

先看下面这个例子,直接在异步方法内部使用了try..catch。

static void Main(string[] args){     Task t = BadAsync();    t.Wait();    Console.WriteLine("Task Status:       {0}", t.Status );    Console.WriteLine("Task IsFaulted:    {0}", t.IsFaulted );    Console.WriteLine("Please enter a key to exit!");    Console.ReadKey(); } static async Task BadAsync(){    try    {        await Task.Run(() => { throw new Exception(); });    }    catch    {        Console.WriteLine("Exception in BadAsync");    }}

执行结果:

Exception in BadAsync
Task Status:       RanToCompletion
Task IsFaulted:    False
Please enter a key to exit!

从结果可以看到,虽然在异步方法内部进行了try..catch,并且也catch到了异常,但是对于调用函数,返回的Task状态依然为 RanToCompletion 。

为什么这个亚子?,原因如下:

  • ① Task没有被取消掉

  • ② 没有未处理的异常。类似的IsFaulted是false。

7.在调用方法中同步的等待任务(WaitAll、WaitAny)

对于单个Task ,可以通过task对象的wait()方法来进行等待。

Task<int> t = CountCharactersAsync("http://www.163.com");t.Wait();

对于多个Task,可以使用WaitAll()或者waitAny()方法,进行同步。

WaitAll是等待所以的任务完成才继续操作

Task<int> t1 = CountCharactersAsync(1, "http://www.163.com");Task<int> t2 = CountCharactersAsync(2, "http://www.microsoft.com");Task<int>[] tasks = new Task<int>[] { t1, t2 };Task.WaitAll(tasks);

WaitAny是只要一个完成就可以继续操作

Task<int> t1 = CountCharactersAsync(1, "http://www.163.com");Task<int> t2 = CountCharactersAsync(2, "http://www.microsoft.com");Task<int>[] tasks = new Task<int>[] { t1, t2 };Task.WaitAny(tasks);

8.在异步方法中异步的等待任务 (WhenAll、.WhenAny)

上面说明了如何在“调用方法”中,同步等待Task的完成。 但是有时候,我们在一个异步方法中也会存在多个任务,想要让它们通过await表达式等待。我们可以通过Task.WhenAll() 和 Task.WhenAny() 方法实现。 这两个方法称为组合子(combinator)。

private async Task<int> CountCharactersAsync(string site1, string site2){    WebClient wc1 = new WebClient();    WebClient wc2 = new WebClient();     Task<string> t1 = wc1.DownloadStringTaskAsync(new Uri(site1));    Task<string> t2 = wc2.DownloadStringTaskAsync(new Uri(site2));     List<Task<string>> tasks = new List<Task<string>>();    tasks.Add(t1);    tasks.Add(t2);     //组合子    await Task.WhenAll(tasks);    //await Task.WhenAny(tasks);     Console.WriteLine("   CCA:  T1 {0} Finished", t1.IsCompleted ? "" : "Not");    Console.WriteLine("   CCA:  T2 {0} Finished", t2.IsCompleted ? "" : "Not");     return t1.IsCompleted? t1.Result.Length: t2.Result.Length;}

9.使用Task.Delay 暂停线程处理

一般我们都使用Thread.Sleep(xxxx) 进行线程的延时,但是 Thread.Sleep会阻塞线程。而Task.Delay则不会阻塞线程,线程可以继续处理其他的工作。

class Simple{    Stopwatch sw = new Stopwatch();    public void DoRun()    {        Console.WriteLine("Caller: Before call");        ShowDelayAsync();        Console.WriteLine("Caller: After call");     }    private async void ShowDelayAsync()    {        sw.Start();        Console.WriteLine("   Before Delay:  {0} ", sw.Elapsed.Milliseconds );        await Task.Delay(1000);        Console.WriteLine("   After  Delay:  {0} ", sw.Elapsed.Milliseconds);    }}

到此,关于“C#怎么使用async和await实现异步编程”的学习就结束了,希望能够解决大家的疑惑。理论与实践的搭配能更好的帮助大家学习,快去试试吧!若想继续学习更多相关知识,请继续关注编程网网站,小编会继续努力为大家带来更多实用的文章!

--结束END--

本文标题: C#怎么使用async和await实现异步编程

本文链接: https://www.lsjlt.com/news/343383.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • C#怎么使用async和await实现异步编程
    这篇文章主要介绍“C#怎么使用async和await实现异步编程”,在日常操作中,相信很多人在C#怎么使用async和await实现异步编程问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”C#怎么使用async...
    99+
    2023-07-02
  • C#使用async和await实现异步编程
    最近在写程序的时候,经常遇到大量需要异步访问的情况,但是对于async和await到底怎么写,还不是非常明确。 1.普通的程序怎么写? class Program { sta...
    99+
    2022-11-13
  • C#异步编程之async/await怎么使用
    今天小编给大家分享一下C#异步编程之async/await怎么使用的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。概述异步这个...
    99+
    2023-07-05
  • .NET怎么实现异步编程async和await
    本篇内容介绍了“.NET怎么实现异步编程async和await”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!await和async是.NET...
    99+
    2023-06-29
  • .NET实现异步编程async和await
    await和async是.NET Framework4.5框架、C#5.0语法里面出现的,await和async是语法糖。 注意: 1、async出现在方法的声明里面,任何一个方法都...
    99+
    2022-11-13
  • C#异步编程async/await用法详解
    异步函数简介 一般指 async 修饰符声明得、可包含await表达式得方法或匿名函数。 声明方式 异步方法的声明语法与其他方法完全一样, 只是需要包含 async 关键字。asyn...
    99+
    2022-11-13
  • C#5.0中的异步编程关键字async和await
    一、Asynchronous methods 异步方法 .NET 4.5 的推出,对于C#又有了新特性的增加——就是C#5.0中async和await两个关键...
    99+
    2022-11-13
  • 使用Node.js的async和await进行异步编程
    Node.JS官方文档:https://nodejs.dev/en/ 创建异步函数,并返回相关数值: 一般方式创建 function fn(){ return Promi...
    99+
    2023-05-18
    Node.js async和await Node.js 异步编程
  • C#异步编程由浅入深(二)之Async/Await的使用
      考虑到直接讲实现一个类Task库思维有点跳跃,所以本节主要讲解Async/Await的本质作用(解决了什么问题),以及Async/Await的工作原理。实现一...
    99+
    2022-11-13
  • 怎么使用c#异步操作async await状态机
    本篇内容介绍了“怎么使用c#异步操作async await状态机”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!async awai...
    99+
    2023-07-05
  • JS中的async与await异步编程及await使用陷阱
    ECMA2017中新加入了两个关键字async与await 简单来说它们是基于promise之上的的语法糖,可以让异步操作更加地简单明了 首先我们需要用async关键字,将函数标记为...
    99+
    2023-03-24
    async与await异步编程 async与await
  • NodeJs中怎么使用async和await处理异步
    NodeJs中怎么使用async和await处理异步,很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。场景远古时代我们在编写exp...
    99+
    2022-10-19
  • C#中使用async和await实现异步Udp通讯的示例代码
    目录C/S架构客户端实现客户端主流程和实现客户端发送消息实现客户端监听消息实现服务器实现服务器主流程和实现服务器发送消息实现服务器监听消息实现总结在之前的C#版本中, 如果我们想要进...
    99+
    2022-11-13
  • js中怎么用async和await实现同步
    这篇“js中怎么用async和await实现同步”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“js中怎么用async和awa...
    99+
    2023-06-29
  • 如何用ES7中的Async和Await进行异步编程
    这篇文章给大家介绍如何用ES7中的Async和Await进行异步编程,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。Async/Await基本规则async 表示这是一个async函数,...
    99+
    2022-10-19
  • JS的异步函数async/await怎么使用
    这篇“JS的异步函数async/await怎么使用”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“JS的异步函数async/a...
    99+
    2023-07-05
  • JS中的async与await异步编程及await使用陷阱源码分析
    这篇“JS中的async与await异步编程及await使用陷阱源码分析”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“JS中...
    99+
    2023-07-05
  • C#中怎么利用AsyncResult实现异步编程
    这篇文章给大家介绍C#中怎么利用AsyncResult实现异步编程,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。C#异步编程模式IAsyncResult概述IAsyncResult 异步设计模式通过名为 BeginOp...
    99+
    2023-06-17
  • C#中怎么实现异步网络编程
    这期内容当中小编将会给大家带来有关C#中怎么实现异步网络编程,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。C#异步网络编程两大方法一、Asynchronous Sockets的方法Socket类的很多连接...
    99+
    2023-06-17
  • Python 中怎么使用Asyncio实现异步编程
    Python 中怎么使用Asyncio实现异步编程,针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。异步是怎么一回事在传统的顺序编程中, 所有发送给解释器的指令会一条条被执行。...
    99+
    2023-06-17
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作