前面一篇文章,了解了任务并行库。这是异步编程基础设施,它允许我们以模块化的方式设计程序,来组合不同的异步操作。
解决了以往线程之间传递消息难等问题,但是我们在阅读和编写此类程序时还是会觉得非常难理解程序的实际执行顺利,总感觉把主动权交给并行库,总觉得心里面没底。而且随着系统的不断扩大,业务的不断完善,了解程序的先后执行次序就变得很重要。
C#5.0
在C#5.0中,微软引入了一个全新的语言特性,异步函数。它是TPL之上的更高级别的抽象,真正简化了异步编程。
抽象隐藏了主要的实现细节,使得程序员无需考虑许多重要的事情,从而使异步编程更加容易。
async Task<string> GetStringAsync() { await Task.Delay(TimeSpan.FromSeconds(2)); return "你好,阿辉"; }
一、async和await
- 在async标识的方法体里面,如果没有await关键字出现,那么这种方法和调用普通的方法没什么区别。
- 在async标识的方法体里面,在await关键字出现之前,还是主线程顺序调用执行,知道await关键字的出现才会出现线程阻塞。
- await关键字可以理解为等待方法执行完毕,除了可以标记有async关键字的方法外还能标记Task对象,标识等待该线程执行完毕。所以await关键字并不针对于asyncd的方法,而是针对async方法所返回给我们的Task。
- async方法的返回类型必须为void、Task或者Task类型,说白了,async要么是void,要么和Task关联。
在async方法外不能使用await关键字,否则编译错误。异步函数在代码中至少要拥有一个await操作符,如果没有,也不会报错,只是会提示警告。
在ASP.NET程序中,一旦在代码中使用了异步,最好一直使用。调用异步方法时,应该用await等待它返回Task对象,一定要避免使用Task.Wait或Task.Result方法,因为这两个方法会阻塞线程。
public ActionResult DeadLock() { Task task = WaitAsync(); //同步程序块,正在等待异步方法完成 task.Wait(); return Content("执行完毕"); } async Task WaitAsync() { await Task.Delay(TimeSpan.FromSeconds(5)); }
在DeadLock()方法中,当执行到task.Wait()方法时,主线程会被阻塞,不往下执行。等候WaitAsync()异步方法执行完毕后,才解除阻塞。
在C#5.0中,await和Async是有一定的限制的。
- 不能把控制台中的Main()方法标记为async.
- 不能在catch、finally、lock或unsafe代码块中使用await操作符。
- 不允许对任何异步函数使用ref或out参数。
并不是所有的方法都得上Async的,对于需要异步处理的该使用async/await就使用,不需要的就使用平常方法。平常的方法在性能方面要比async关键字方法块40~50倍。所以我们在平时也要注意不能滥用async。
二、使用await操作符获取异步任务结果
async static Task WithWait() { try { string result = await GetInfoAsync("阿辉"); Console.WriteLine(result); } catch (Exception ex) { Console.WriteLine(ex.Message); } } async static Task<string> GetInfoAsync(string name) { await Task.Delay(TimeSpan.FromSeconds(2)); return name+",你好。"; } Task t = WithWait(); t.Wait();
这样就可以获取到异步任务的返回值了。使用await后,C#立即创建了一个任务,其有一个后续操作任务,包含了await操作符后面的所有剩余代码。这个新任务也处理了异常传播。然后将任务返回到主方法中并等待其完成。
目前在最新的.NET Core中,基本上所有方法都默认在使用await/async这种模式来编写。所以学习这种异步编程方式还是特别有意义的。
好了,今天的学习就到这里,我们下篇文章见。