C#5.0中的异步编程关键字async和await

 更新时间:2022年06月09日 10:06:22   作者:springsnow  
这篇文章介绍了C#5.0中的异步编程关键字async和await,文中通过示例代码介绍的非常详细。对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下

一、Asynchronous methods 异步方法

.NET 4.5 的推出,对于C#又有了新特性的增加——就是C#5.0中async和await两个关键字,这两个关键字简化了异步编程。

  • 使用async修饰的方法被称为异步方法,这个方法调用时应该在前面加上await。
  • 异步方法命名应该以Async结尾,这样大家知道调用的时候使用await。

async和await关键字只是编译器的功能,编译器最终会用Task类创建代码。

1、创建返回任务的异步方法

建立一个同步方法Greeting,该方法在等待一段时间后,返回一个字符串。

private string Greeting(int delay, string name)
{
    System.Threading.Thread.Sleep(delay);
    return string.Format("Hello, {0}.", name);
}

定义一个方法GreetingAsync,可以使方法异步化,其传入的参数不做强制要求。

异步方法,返回类型必须为Task、Task或void。不能作为程序的入口点,即Main方法不能使用async修饰符。

基于任务的异步模式指定,并返回一个任务。

注意,该方法返回的是Task,定义了一个返回字符串的任务,与同步方法返回值一致。

private Task<string> GreetingAsync(string name, int delay = 3000)
{   
    return Task.Run<string>(() =>
    {
        return Greeting(delay, name);
    });
}

2、调用异步方法

可以使用await关键字调用返回任务的异步方法GreetingAsync。

注意:await修饰符只能用于返回Task或者Task的方法。

并且使用await关键字的方法,这里是CallerWithAsync(),必须要用async关键字修饰符声明。

在GreetingAsync方法完成前,被async关键字修饰的方法内await关键字后面的代码不会继续执行。

但是,启动被async关键字修饰的方法的线程可以被重用,而没有被阻塞。

public async void CallerWithAsync()
{   
    string result = await GreetingAsync("Nigel", 2000);
    Console.WriteLine(result);
}

3、简单实例

void Main()
{
    DisplayValue();
    System.Diagnostics.Debug.WriteLine("MyClass() End.");
}

public async void DisplayValue()
{
    double result = await GetValueAsync(1234.5, 1.01);//此处会开新线程处理GetValueAsync任务,然后方法马上返回。这之后的所有代码都会被封装成委托,在GetValueAsync任务完成时调用
    System.Diagnostics.Debug.WriteLine("Value is : " + result);
}

public Task<double> GetValueAsync(double num1, double num2)
{
    return Task.Run(() =>
    {
        for (int i = 0; i < 1000000; i++)
        {
            num1 = num1 / num2;
        }
        return num1;
    });
}

上面在MyClass的构造函数里调用了async关键字标记的异步方法DisplayValue(),DisplayValue()方法里执行了一个await关键字标记的异步任务GetValueAsync(),这个异步任务必须是以Task或者Task作为返回值的。

而我们也看到,异步任务执行完成时实际返回的类型是void或者TResult,DisplayValue()方法里await GetValueAsync()之后的所有代码都会在异步任务完成时才会执行。

DisplayValue()方法实际执行的代码如下:

public void DisplayValue()
{
    System.Runtime.CompilerServices.TaskAwaiter<double> awaiter = GetValueAsync(1234.5, 1.01).GetAwaiter();
    awaiter.OnCompleted(() =>
    {
        double result = awaiter.GetResult();
        System.Diagnostics.Debug.WriteLine("Value is : " + result);
    });
}

可以看到,async和await关键字只是把上面的代码变得更简单易懂而已。

程序的输出如下:

MyClass() End.

Value is : 2.47032822920623E-322

4、使用async 和await定义异步方法不会创建新线程, 它运行在现有线程上执行多个任务。

// 使用C# 5.0中提供的async 和await关键字来定义异步方法
// 从代码中可以看出C#5.0 中定义异步方法就像定义同步方法一样简单。
private async Task<long>  AccessWebAsync()
{
    MemoryStream content = new MemoryStream();

    // 对MSDN发起一个Web请求
    HttpWebRequest webRequest = WebRequest.Create("http://msdn.microsoft.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;
}

运行结果如下:

三、async和await关键字剖析

我们对比下上面使用async和await关键字来实现异步编程的代码和在第二部分的同步代码,有没有发现使用async和await关键字的异步实现和同步代码的实现很像,只是异步实现中多了async和await关键字和调用的方法都多了async后缀而已。

正是因为他们的实现很像,所以我在第四部分才命名为使用async和await使异步编程更简单,就像我们在写同步代码一样,并且代码的coding思路也是和同步代码一样,这样就避免考虑在APM中委托的回调等复杂的问题,以及在EAP中考虑各种事件的定义。

下面再分享下几个关于async和await常问的问题

  • 问题一:是不是写了async关键字的方法就代表该方法是异步方法,不会堵塞线程呢?

答: 不是的,对于只标识async关键字的(指在方法内没有出现await关键字)的方法,调用线程会把该方法当成同步方法一样执行,所以然而会堵塞GUI线程,只有当async和await关键字同时出现,该方法才被转换为异步方法处理。

  • 问题二:“async”关键字会导致调用方法用线程池线程运行吗?

答: 不会,被async关键字标识的方法不会影响方法是同步还是异步运行并完成,而是,它使方法可被分割成多个片段,其中一些片段可能异步运行,这样这个方法可能异步完成。这些片段界限就出现在方法内部显示使用”await”关键字的位置处。所以,如果在标记了”async”的方法中没有显示使用”await”,那么该方法只有一个片段,并且将以同步方式运行并完成。在await关键字出现的前面部分代码和后面部分代码都是同步执行的(即在调用线程上执行的,也就是GUI线程,所以不存在跨线程访问控件的问题),await关键处的代码片段是在线程池线程上执行。总结为——使用async和await关键字实现的异步方法,此时的异步方法被分成了多个代码片段去执行的,而不是像之前的异步编程模型(APM)和EAP那样,使用线程池线程去执行一整个方法。

到此这篇关于C#5.0异步编程之async和await的文章就介绍到这了。希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • C#实现文件与二进制互转并存入数据库

    C#实现文件与二进制互转并存入数据库

    这篇文章主要介绍了C#实现文件与二进制互转并存入数据库,本文直接给出代码实例,代码中包含详细注释,需要的朋友可以参考下
    2015-06-06
  • 在Unity中使用全局变量的操作

    在Unity中使用全局变量的操作

    这篇文章主要介绍了在Unity中使用全局变量的操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2021-04-04
  • C#全角半角转换函数代码分享

    C#全角半角转换函数代码分享

    这篇文章介绍了C#全角半角转换函数代码,有需要的朋友可以参考一下
    2013-09-09
  • C#利用KPM算法解决字符串匹配问题详解

    C#利用KPM算法解决字符串匹配问题详解

    Knuth-Morris-Pratt 字符串查找算法,简称为 “KMP算法”,常用于在一个文本串S内查找一个模式串P 的出现位置。本文将利用该算法解决字符串匹配问题,感兴趣的可以学习一下
    2022-11-11
  • C#使用SQL DataReader访问数据的优点和实例

    C#使用SQL DataReader访问数据的优点和实例

    今天小编就为大家分享一篇关于C#使用SQL DataReader访问数据的优点和实例,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧
    2018-10-10
  • C# 线程同步的方法

    C# 线程同步的方法

    这篇文章主要介绍了C# 线程同步的方法,文中讲解非常细致,代码帮助大家更好的理解和学习,感兴趣的朋友可以了解下
    2020-08-08
  • C#调用系统API指定快捷键的方法

    C#调用系统API指定快捷键的方法

    这篇文章主要介绍了C#调用系统API指定快捷键的方法,涉及C#快捷键的操作技巧,需要的朋友可以参考下
    2015-06-06
  • 浅谈C#9.0新特性之参数非空检查简化

    浅谈C#9.0新特性之参数非空检查简化

    这篇文章主要介绍了浅谈C#9.0新特性之参数非空检查简化,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-06-06
  • c#之关闭窗体的实现方法

    c#之关闭窗体的实现方法

    这篇文章主要介绍了c#之关闭窗体的实现方法,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-07-07
  • C# datatable 不能通过已删除的行访问该行的信息处理方法

    C# datatable 不能通过已删除的行访问该行的信息处理方法

    采用datatable.Rows[i].Delete()删除行后再访问该表时出现出现“不能通过已删除的行访问该行的信息”的错误
    2012-11-11

最新评论