我目前正在尝试在必须以.Net 4.0为目标的程序中实现.Net 4.5的Task.Delay()方法的替代品.我在
this blog找到了以下代码.
- /* You can write Task-based asynchronous methods by utilizing a TaskCompletionSource.
- A TaskCompletionSource gives you a 'slave' Task that you can manually signal.
- Calling SetResult() signals the task as complete,and any continuations kick off. */
- void Main()
- {
- for (int i = 0; i < 10000; i++)
- {
- Task task = Delay (2000);
- task.ContinueWith (_ => "Done".Dump());
- }
- }
- Task Delay (int milliseconds) // Asynchronous NON-BLOCKING method
- {
- var tcs = new TaskCompletionSource<object>();
- new Timer (_ => tcs.SetResult (null)).Change (milliseconds,-1);
- return tcs.Task;
- }
任务对我来说都是新手. System.Threading.Timer和TaskCompletionSource对我来说是全新的(截至今天),我正在与他们挣扎.除此之外,我想知道如何在此代码中添加CancellationToken功能.我假设我可以像这样在Delay()方法中添加一个参数:
- Task Delay (int milliseconds,CancellationToken token) // Asynchronous NON-BLOCKING method
- {
- var tcs = new TaskCompletionSource<object>();
- new Timer (_ => tcs.SetResult (null)).Change (milliseconds,-1);
- return tcs.Task;
- }
解决方法
我试图尽可能少地更改你的代码,但这是一个工作示例,其行为与Task.Delay相同.
重要的是要注意我使用TrySetCanceled和TrySetResult,因为Timer可以在任务取消后完成.理想情况下,您想要停止计时器.
另请注意,取消的任务将抛出TaskCanceledException
- static void Main(string[] args)
- {
- // A cancellation source that will cancel itself after 1 second
- var cancellationTokenSource = new CancellationTokenSource(TimeSpan.FromSeconds(1));
- try
- {
- // This will only wait 1 second because as it will be cancelled.
- Task t = Delay(5000,cancellationTokenSource.Token);
- t.Wait();
- Console.WriteLine("The task completed");
- }
- catch (AggregateException exception)
- {
- // Expecting a TaskCanceledException
- foreach (Exception ex in exception.InnerExceptions)
- Console.WriteLine("Exception: {0}",ex.Message);
- }
- Console.WriteLine("Done");
- Console.ReadLine();
- }
- private static Task Delay(int milliseconds,CancellationToken token)
- {
- var tcs = new TaskCompletionSource<object>();
- token.Register(() => tcs.TrySetCanceled());
- Timer timer = new Timer(_ => tcs.TrySetResult(null));
- timer.Change(milliseconds,-1);
- return tcs.Task;
- }
再读一下你的问题.如果您需要Task.Delay并且您的目标是.NET 4.0,那么您应该使用http://www.nuget.org/packages/Microsoft.Bcl.Async/中的Microsoft Async nuget包它包含方法TaskEx.Delay