虽然你可以依靠WithCancellation
再利用的目的,超时简单的解决方案(不扔OperationCanceledException
)是创建一个超时任务与Task.Delay
,等待第一个任务使用Task.WhenAny
完成:
public static Task<TResult> WithTimeout<TResult>(this Task<TResult> task, TimeSpan timeout)
{
var timeoutTask = Task.Delay(timeout).ContinueWith(_ => default(TResult), TaskContinuationOptions.ExecuteSynchronously);
return Task.WhenAny(task, timeoutTask).Unwrap();
}
或者,如果您想在发生超时而不是仅仅返回默认值的情况下抛出异常(即, null
):
public static async Task<TResult> WithTimeout<TResult>(this Task<TResult> task, TimeSpan timeout)
{
if (task == await Task.WhenAny(task, Task.Delay(timeout)))
{
return await task;
}
throw new TimeoutException();
}
而且用法是:
var content = await Response.Content.ReadAsStringAsync().WithTimeout(TimeSpan.FromSeconds(1));
仅供参考,如果你处理的IDisposable的超时你可能想看看这个:[异步网络运营写不完](HTTP ://stackoverflow.com/a/21468138/885318) – i3arnon 2014-09-23 06:09:07