有Task.Delay.NET 4.5中
在.NET 4.0中我该如何做?
有Task.Delay.NET 4.5中
在.NET 4.0中我该如何做?
Thread.Sleep
在你的代码几乎总是一个错误。
Answers:
您可以使用在4.0中Timer
创建Delay
方法:
public static Task Delay(double milliseconds)
{
var tcs = new TaskCompletionSource<bool>();
System.Timers.Timer timer = new System.Timers.Timer();
timer.Elapsed+=(obj, args) =>
{
tcs.TrySetResult(true);
};
timer.Interval = milliseconds;
timer.AutoReset = false;
timer.Start();
return tcs.Task;
}
Callback()
定义上使用Dispatcher.BeginInvoke()
?
TrySetResult
永远不会被调用。
Timer
该类专门在内部进行处理,以确保其用户在其生命周期内无需保留对其的引用。只要计时器当前正在运行,它就会从有根的位置添加对自身的引用,然后在不再运行时将其删除。
使用来自NuGet的Microsoft.Bcl.Async程序包,它具有TaskEx.Delay
。
using System;
using System.Threading;
using System.Threading.Tasks;
class Program
{
static void Main()
{
Delay(2000).ContinueWith(_ => Console.WriteLine("Done"));
Console.Read();
}
static Task Delay(int milliseconds)
{
var tcs = new TaskCompletionSource<object>();
new Timer(_ => tcs.SetResult(null)).Change(milliseconds, -1);
return tcs.Task;
}
}
在“如何在4.0中实现Task.Delay ”部分中
object
的值打印ToString
出来。请注意,他没有在实际实现中使用该方法,也没有使用任何其他非库方法,而只是对其进行测试的示例函数。
以下是可取消Task.Delay实现的代码和示例工具。您可能对该Delay
方法感兴趣:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace DelayImplementation
{
class Program
{
static void Main(string[] args)
{
System.Threading.CancellationTokenSource tcs = new System.Threading.CancellationTokenSource();
int id = 1;
Console.WriteLine(string.Format("Starting new delay task {0}. This one will be cancelled.", id));
Task delayTask = Delay(8000, tcs.Token);
HandleTask(delayTask, id);
System.Threading.Thread.Sleep(2000);
tcs.Cancel();
id = 2;
System.Threading.CancellationTokenSource tcs2 = new System.Threading.CancellationTokenSource();
Console.WriteLine(string.Format("Starting delay task {0}. This one will NOT be cancelled.", id));
var delayTask2 = Delay(4000, tcs2.Token);
HandleTask(delayTask2, id);
System.Console.ReadLine();
}
private static void HandleTask(Task delayTask, int id)
{
delayTask.ContinueWith(p => Console.WriteLine(string.Format("Task {0} was cancelled.", id)), TaskContinuationOptions.OnlyOnCanceled);
delayTask.ContinueWith(p => Console.WriteLine(string.Format("Task {0} was completed.", id)), TaskContinuationOptions.OnlyOnRanToCompletion);
}
static Task Delay(int delayTime, System.Threading.CancellationToken token)
{
TaskCompletionSource<object> tcs = new TaskCompletionSource<object>();
if (delayTime < 0) throw new ArgumentOutOfRangeException("Delay time cannot be under 0");
System.Threading.Timer timer = null;
timer = new System.Threading.Timer(p =>
{
timer.Dispose(); //stop the timer
tcs.TrySetResult(null); //timer expired, attempt to move task to the completed state.
}, null, delayTime, System.Threading.Timeout.Infinite);
token.Register(() =>
{
timer.Dispose(); //stop the timer
tcs.TrySetCanceled(); //attempt to mode task to canceled state
});
return tcs.Task;
}
}
}
这是一个基于计时器的简洁实现,并进行了适当的清理:
var wait = new TaskCompletionSource<bool>();
using (new Timer(_ => wait.SetResult(false), null, delay, Timeout.Infinite))
await wait.Task;
要在.NET 4.0上使用此代码,您需要Microsoft.Bcl.Async NuGet程序包。
await
关键字是C#语言,这是从.NET Framework版本分开的一部分。问题是,Task<T>
没有GetAwaiter
,这await
依赖。Microsoft.Bcl.Async提供了这一点。我更新了答案以提及NuGet软件包。
TaskEx.Delay
更简洁吗?