Windows 8 系列(七):使用异步API:await 和 async
Windows 8 Metro 编程相对于Windows phone 和 Silverlight 最大的不同点之一就是大量的使用了异步API,而且这些API通常没有相对应的同步API,因此我们需要确保了解如何在Metro风格应用中使用异步API。
使用异步API时不可避免的会碰到两个关键词:await 和 async,下面我们举例说明异步API的优势和具体的用法。
异步的API一般是需要较长时间才能返回结果或者完成的(例如:I.O.操作)。假设我们在点击一个Button之后,程序会读取一个比较大的本地文件。
1. 如果用同步的方式读取,那么在返回读取结果之前的这段时间内,你得程序不能同时进行其他相应,比如用户再次的点击,也就是说你得程序是阻塞状态。
2. 如果换成异步操作就不同了,程序在执行到这一行代码时,应用会临时退出时间处理程序,它可以在等待返回结果的过程中同时处理其他的用户相应,直到成功返回结果后继续进行下面的代码。
(经Damon.Tian提醒,这里代码做了修改,返回类型应为Task<bool>)
private async Task<bool> IsSuccess (string fileName, StorageFolder folder) { var file = await folder.CreateFileAsync(fileName, CreationCollisionOption.OpenIfExists); string tempString = "Wow!";
return true; }
这就是读取文件的API,在这里使用的await关键字,代表调用异步接口并等待返回结果后才执行后面的代码(记住这句话,后面会讲到)。
再看看这个函数在定义时用了一个关键字:async,这是因为在此函数中至少有一个地方用了await关键字,那么这个函数就必须定义为async。在外面调用这个函数的时候也就可以用await关键字了,例如:
bool result = await IsSuccess(fileName, folder);
同样的,上面这行代码所在的函数,也必须声明async关键字。
接下来,问题来了,这两个关键字是不是必须要写的呢?可不可以不用?答案是,可以。
如果在调用异步API时没有用到await关键字,也是可以的。
我们还拿上面的function举例,如果写成下面这样也是可以的,但是在编译的时候会生成一个警告:没有用await关键字。
var file = folder.CreateFileAsync(fileName, CreationCollisionOption.OpenIfExists);
那这两种写法有什么不同呢?答案是:代码的执行顺序有了改变,而且返回结果的类型也变了。
如果调用异步API时没有用await,那么在执行异步操作的时候,程序是不会等待API返回结果再执行下面的代码,而是直接执行下面的代码:
也就是说,如果代码变成下面这个样子,那么file还没有被成功赋值时(CreateFileAsync函数还未返回结果时),tempString就已经被赋值了。
(如果不带await 这里返回的file不再是StorageFile类型而是 IAsyncOperation<StorageFile>类型)
private async Task<bool> IsSuccess (string fileName, StorageFolder folder) { var file = folder.CreateFileAsync(fileName, CreationCollisionOption.OpenIfExists); string tempString = "Wow!";
return true; }
例如:应用中一般都会有弹出框弹出警告,例如:假设我们写一个demo,发布博文成功后用弹出系统警告框,下面是公共函数:弹出系统警告框
public async static void Alert(string message, string title = "") { MessageDialog dlg = new MessageDialog(message, title); await dlg.ShowAsync(); }
代码一:
Alert("您还未登陆。"); Alert("请登录!");
代码二:
await Alert("您还未登陆。"); await Alert("请登录!");
以上两段代码执行的结果不同,第一段代码会连续弹出两个弹出框,先弹出“您还未登录。”,然后弹出“请登录!”,第二个弹出框在第一个弹出框的上面。
代码二会先弹出第一个警告:未登录,用户点击“确定”按钮关闭弹出框后,才会弹出第二个弹出框。
另外需要注意的是:
await无法用在类似lock块代码中:
lock(tempString) { //这里不能用await }
Lamda表达式可以这样写
this.Loaded += async (sender, args) =>
{
await ...
};
以上是我的个人简介,有任何问题或不同建议,欢迎提出,大家一起讨论,更好的铺平win8开发之路!
我的微博:http://weibo.com/345169632 介是为什么