主要内容 3种不同模式的异步编程:异步编程、基于事件的异步编程、基于任务的异步编程(TAP)
学习重点 TAP 的关键字 async await
一、异步模式
传统的异步模式可以分为两种,一是,对于一个同步方法如XXX(),其对应的异步模式版本为BeginXXX()和EndXXX()。BeginXXX()接受所有同步方法的输入参数,并且定义了一个AsyncCallback参数,用于在异步方法执行完成后调用委托;EndXXX()使用所有同步方法的输出参数,按照同步方法返回类型返回结果。
二是利用委托的BeginInvoke方法和EndInvoke方法,简单的说从BeginInvoke()调用开始,创建一个新的线程,前台线程继续执行,EndInvoke()会阻塞线程,知道委托的方法执行完毕。
1.两种显示程序运行的方法
IAsyncResult result = newTask.BeginInvoke(10000, null, null);
while (!result.IsCompleted)//方法一
{
Thread.Sleep(1000);
Console.Write("*");
}
while (!result.AsyncWaitHandle.WaitOne(100))//方法二
{
Console.Write("*");
}
int n = newTask.EndInvoke(result);
2.BeginInvoke的输入参数和返回值
参数可以分为两部分,第一部分是调用方法的输入参数,第二部分是回调AsyncCallback,回调方法必须接受一个IAsyncResult参数,接下来的状态参数可以传递委托,从而实现在外部调用EndInvoke()
if (result == null) return;
string str1 = (result.AsyncState as MyMethod).EndInvoke(result).ToString();
二、TAP
1.下面的代码说明了同步方法向异步方法的改造
public static string Greeting(string name)
{
Task.Delay(5000).Wait();//Delay也返回一个任务
return $"Hello,{name}";
}
public static Task<string> GreetingAsync(string name)
{
return Task.Run<string>(() =>
{
return Greeting(name);
});
}
Run<string>要求传递Func<string>委托方法,因为GreetingAsync需要返回Task<string>
2. 注意使用关键字asyncawait ,我的理解是,需要使用await的方法,都需要async,除了特殊的函数,如main;还有一点,await只能修饰返回task的函数;
另外await所在函数里在其后的代码不再执行直到任务完成,因此单独定义一个异步函数,并用async修饰。
private async static void CallerWithAsync()
{
string result = await GreetingAsync("huangxianxiang");
WriteLine(result);
}
3.await延续任务 下面代码可以看出await后面应该避免使用ui元素,避免切换到同步上下文会执行的更快
private static void CallerWithContinuousTask()
{
Task<string> t1 = GreetingAsync("huangxianxiang");
t1.ContinueWith(t =>
{
string result = t.Result;
WriteLine(result);
});
}
4.组合器
Task<string> s1 = GreetingAsync("sunyan");
Task<string> s2 = GreetingAsync("huangxianxiang");
string[] strList = await Task.WhenAll(s1, s2);