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 介是为什么

我的博客:http://www.cnblogs.com/lihaiyin

posted @ 2012-08-16 11:45  lihaiyin  阅读(3503)  评论(8编辑  收藏  举报