本文共 1845 字,大约阅读时间需要 6 分钟。
前文传送门 dotNET开发系列
之前在实际工作中,遇到过这样的问题,异步获取 GPS 定位信息。一个实际的问题出现了,在第一次定位的时候一般时间都比较长,现在的要做的是,当超出一定的时间,就提醒用户。
那么如何实现异步等待任务并在超时时进行一定的处理呢?
public bool Wait(int millisecondsTimeout, CancellationToken cancellationToken);public bool Wait(int millisecondsTimeout);public bool Wait(TimeSpan timeout);public void Wait();public void Wait(CancellationToken cancellationToken);
Wait 方法所支持的的特点:一个是取消任务,一个是超时。但是不可避免的是阻塞线程,一般在主线程中(UI 线程)是要避免耗时任务的。如果真要去等待获取定位信息的 Task,会不可避免地卡住界面,用户体验不够良好。
Task.When** 可等待多个异步任务,不阻塞线程,所以可以利用 Delay 静态方法“间接”实现异步超时的处理,非阻塞的方式!
var resultTask = Task.WhenAny(task, Task.Delay(timeout))
我们再 WhenAny 方法中代入两个 task 参数:源 task、新建的延迟的 task(ps:即超时的时间 task),再得到 resultTask。
// 摘要:任何提供的任务已完成时,创建将完成的任务。// 参数:tasks:等待完成的任务。// 返回结果:表示提供的任务之一已完成的任务。返回任务的结果是完成的任务。public static TaskWhenAny(params Task[] tasks);
Task.WhenAny(task, Task.Delay(timeout)),再比较这两个 Task 的执行先后。实现代码如下:
public static async TaskWaitAsync (Task task, TimeSpan timeout) { if (await Task.WhenAny(task, Task.Delay(timeout)) == task) { //指定时间内完成的处理 return await task; } else { //超时处理 throw new TimeoutException("The operation has timed out."); } }
上述方法所实现的等待一个获取 gps 的耗时任务,是没有任何问题的。并不会产生性能、卡界面等问题,但是如果在源 task(ps:执行获取 gps 的 task),在超时到期之前完成,则不会取消在 Task.Delay 调用中启动的内部计时器作业。当调用多次该方法时,“僵尸”计时器作业的数量变得越来越大时。性能可能会受到影响
https://devblogs.microsoft.com/pfxteam/crafting-a-task-timeoutafter-method/
https://stackoverflow.com/questions/4238345/asynchronously-wait-for-taskt-to-complete-with-timeout
[1]
Task.ContinueWith: http://msdn.microsoft.com/en-us/library/dd270696.aspx
[2]Task.Wait: http://msdn.microsoft.com/en-us/library/dd235606.aspx
面试
汇总
原创
作死
工具
技巧
转载地址:http://emgbi.baihongyu.com/