async关键字和await是配套使用的异步方法语法糖,配合Task类可以使多线程变得有序,
也可以自己实现一套协程功能。
async关键字(异步方法)
async关键字是C#特有的。Java没有这玩意。
async在C#世界里是上下文关键字。它只有在修饰一个方法的时候才自动被编译器识别为
关键字,在代码的其他位置上可以被用作变量名等其他任何用途。
asyn关键字用来修饰两类方法:lambda表达式或则异步方法。
拥有async修饰的方法称为async方法,比如:
Public Task ExampleMethodAsync()
{
//(1)To do some code here synchronously…(在这里同步做一些代码…)
await…//(2)To do something asynchronously.(异步地做某事。)
//(3)To do some code here after awiat code.(在awiat代码之后做一些代码。)
}
就如上⾯这个⽅法ExampleMethodAsync(),微软爷爷特别喜欢在定义异步函数名字后习
惯加个Async后缀(这不是必须的,加不加编译器既不会报错,也不会影响异步特性),告
诉我们这个⽅法是个异步⽅法。我们在⾃⼰定义异步⽅法的时候,也可以照搬这个微软
的习惯。
asyncawaitawait修饰的⽅法内部,应当出现⼀个关键字,两个关键字⼀般成对出现。当
然,如果我们不⼩⼼忘记写表达式或者语句,这asyncawaitasyncawait个⽅法默认按照同
步⽅式运⾏,同时,编译器会友好地提⽰我们是不是漏写了。此外,⽅法内部,可以有
多个语句。
awiat运⾏的语句,⼀般都是⽐较费时的任务(也就是会阻塞主线程的⼀些操作,⽐如获
取Http应答,写⼊⽂档,保存数据库等),要不然就不需要异步了。
以上⾯的例⼦为例(假设例⼦中的是第⼀个),异步⽅法执⾏过程(⽐较粗的看):
1.主线程进⼊⽅法ExampleMethodAsync()后,先顺序执⾏(1):如果(1)当中有创建Task或
Task的语句,或者是调用其他async方法(返回值是Task或者Task),为了描述⽅便,我们都称为Task创建语句,⽐如直接创建⼀个Task或
Task:
var tsk= Task.Run(() =>
{
Thread.Sleep(1000) ;
Console.Writeline(“异步执行另一个任务。”);
});
或者调⽤另外⼀个async⽅法
Task getStringTask =
client.GetStringAsync(“https://docs.microsoft.com/dotnet”)
那么,在调⽤完Task创建语句的时候异步任务就已经开始运⾏了(这个语句调⽤本⾝是
在主线程当中,内部的任务则是新的线程中执⾏),也就是此时异步的线程已经启动
了,由于它是异步启动的,所以它并不会阻⽌主线程继续往下⾛;
2.接下来,主线程会顺序运⾏到async⽅法内部的第⼀个await,如果第⼀个await调⽤的
仍然是⼀个async⽅法,那么主程序继续进⼊这个⽅法执⾏,⼀直到碰到⼀个await task
为⽌,主线程才会跳出ExampleMethodAsync⽅法;
3.ExampleMethodAsync()⽅法中剩余的(3)在执⾏完await(2)部分的内容才执⾏。
4.假设ExampleMethodAsync()中有第⼆个,第三个…awiat,因为主程序已经跳出来了,
后续的await会在异步线程中按顺序执⾏下去。
async⽅法可以是下⾯三种返回类型:
Task
Task
void 这种返回类型⼀般⽤在event事件处理器中%2C或者⽤在你只需要任务执⾏,不
关⼼任务执⾏结果的情况当中。
任何其他具有GetAwaiter⽅法的类型(从C#7.0开始)