如何使用Timer替换Azure辅助角色中的Thread.Sleep(...)?

时间:2013-02-05 12:34:44

标签: c# .net azure timer azure-worker-roles

有很多人说在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();作为解决方案。这不是我想在这里实现的。请注意帖子标题!

1 个答案:

答案 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;
        }
    }
}