浅谈Async和Await如何简化异步编程(几个实例让你彻底明白)

2019-12-30 15:22:57于海丽


private async void btnClick_Click(object sender, EventArgs e)
    {
      long length = await AccessWebAsync();

      // 这里可以做一些不依赖回复的操作
      OtherWork();

      this.richTextBox1.Text += String.Format("n 回复的字节长度为: {0}.rn", length);
      txbMainThreadID.Text = Thread.CurrentThread.ManagedThreadId.ToString();
    }

    // 使用C# 5.0中提供的async 和await关键字来定义异步方法
    // 从代码中可以看出C#5.0 中定义异步方法就像定义同步方法一样简单。
    // 使用async 和await定义异步方法不会创建新线程,
    // 它运行在现有线程上执行多个任务.
    // 此时不知道大家有没有一个疑问的?在现有线程上(即UI线程上)运行一个耗时的操作时,
    // 为什么不会堵塞UI线程的呢?
    // 这个问题的答案就是 当编译器看到await关键字时,线程会
    private async Task<long> AccessWebAsync()
    {
      MemoryStream content = new MemoryStream();

      // 对MSDN发起一个Web请求
      HttpWebRequest webRequest = WebRequest.Create("http://www.easck.com/zh-cn/") as HttpWebRequest;
      if (webRequest != null)
      {
        // 返回回复结果
        using (WebResponse response = await webRequest.GetResponseAsync())
        {
          using (Stream responseStream = response.GetResponseStream())
          {
            await responseStream.CopyToAsync(content);
          }
        }
      }

      txbAsynMethodID.Text = Thread.CurrentThread.ManagedThreadId.ToString() ;
      return content.Length;
    }

    private void OtherWork()
    {
      this.richTextBox1.Text += "rn等待服务器回复中.................n";
    }

运行结果如下:

async,await,c#,async异步编程,异步

async和await关键字剖析

我们对比下上面使用async和await关键字来实现异步编程的代码和在第二部分的同步代码,有没有发现使用async和await关键字的异步实现和同步代码的实现很像,只是异步实现中多了async和await关键字和调用的方法都多了async后缀而已。正是因为他们的实现很像,所以我在第四部分才命名为使用async和await使异步编程更简单,就像我们在写同步代码一样,并且代码的coding思路也是和同步代码一样,这样就避免考虑在APM中委托的回调等复杂的问题,以及在EAP中考虑各种事件的定义。

从代码部分我们可以看出async和await的使用确实很简单,我们就如在写同步代码一般,但是我很想知道编译器到底给我们做了怎样的处理的?并且从运行结果可以发现,运行异步方法的线程和GUI线程的ID是一样的,也就是说异步方法的运行在GUI线程上,所以就不用像APM中那样考虑跨线程访问的问题了(因为通过委托的BeginInvoke方法来进行回调方法时,回调方法是在线程池线程上执行的)。下面就用反射工具看看编译器把我们的源码编译成什么样子的: