有很多人说在Timer
中使用Thread.Sleep(...)
代替Azure Worker Role
。没有问题。
我正在努力理解的是如何编写代码。
目前,我有以下(pseduo代码)
_timer.Elapsed += (sender, args) => DoWork();
public override void Run()
{
while(true)
{
DoWork();
}
}
public void DoWork()
{
try
{
_timer.Stop();
// Now - do stuff ....
}
catch(....) { ... }
_timer.Start()
}
接下来发生的是,代码进入DoWork()
方法一次,DoesStuff(tm)
..罚款..启动计时器(比如..间隔30秒),然后退出该方法。
然后,它返回到该循环中的主Run()
方法..所以立即回来再次进入DoWork()
方法..而不是等待计时器将其关闭。
所以我不确定如何用计时器替换任何Thread.Sleep(...)
。
任何线索?
我不想退出Run()
方法:)我很高兴永远保持循环。我坚持使用的是替换标准Thread.Sleep(...)
调用(阻止线程)并将其替换为大多数人建议的Timer
。
请不要链接或建议我使用cancelSource.Token.WaitHandle.WaitOne();
作为解决方案。这不是我想在这里实现的。请注意帖子标题!
答案 0 :(得分:6)
我认为如果你想按照这里的大纲来解决这种情况,你需要一个WaitHandle和一个Timer。
简短的回答如下。长篇答案成为博文:HowTo wait in a WorkerRole using Timer and EventWaitHandle over Thread.Sleep
我使用了EventWaitHandle和Timer,并提出了这个解决方案:
public class WorkerRole : RoleEntryPoint
{
Waiter waiter;
public override bool OnStart()
{
waiter = new Waiter(WorkerConfiguration.WaitInterval);
return base.OnStart();
}
public override void Run()
{
while (true)
{
DoWork();
waiter.Wait();
}
}
public void DoWork()
{
// [...]
}
}
这是服务员班:
public class Waiter
{
private readonly Timer timer;
private readonly EventWaitHandle waitHandle;
public Waiter(TimeSpan? interval = null)
{
waitHandle = new AutoResetEvent(false);
timer = new Timer();
timer.Elapsed += (sender, args) => waitHandle.Set();
SetInterval(interval);
}
public TimeSpan Interval
{
set { timer.Interval = value.TotalMilliseconds; }
}
public void Wait(TimeSpan? newInterval = null)
{
SetInterval(newInterval);
timer.Start();
waitHandle.WaitOne();
timer.Close();
waitHandle.Reset();
}
private void SetInterval(TimeSpan? newInterval)
{
if (newInterval.HasValue)
{
Interval = newInterval.Value;
}
}
}