定时器间隔为数月

时间:2010-11-26 01:04:37

标签: c# timer

我有一个Windows服务处理由System.Timers.Timer触发的事件。 我想将该计时器的间隔设置为3个月。

System.Timers.Timer的Interval属性是以毫秒为单位的Int32,而Int32.MaxValue小于3个月(以毫秒为单位)。

我该怎么办?

5 个答案:

答案 0 :(得分:10)

你会重新考虑你的设计。下次要执行事件时存储(例如注册表,文件,数据库......),然后定期唤醒以检查是否已经过了该时间。即使您可以将System.Timers.Timer设置为3个月,系统也可能会在计时器关闭之前重新启动并且您将丢失事件。

另一种选择是使用Windows Scheduler执行的预定作业。它会运行一个小程序,向您的服务发送一条消息,说明事件已经发生。这比资源密集程度更低 - 虽然更复杂 - 而不是定期醒来检查3个月是否已经过去。

答案 1 :(得分:1)

您无法连续运行PC 3个月。如果应用程序关闭,计时器也将关闭。在重新启动应用程序时,计时器从开始重新启动。

如果您想以3个月的间隔发射事件,则必须保留有关退出应用程序时计时器所用总时间的信息。

<强>更新

所以你必须在某些部分划分你的间隔,并在每一次过程中设置一个计数器并增加它。每过一次检查它,直到达到3个月。

<强> e.g。

    int interval = 10 * 24 * 60 * 60 * 1000; // Time interval for 10 days 
    int counter = 0;
    private void timer1_Elapsed(object sender, System.Timers.ElapsedEventArgs e)
    {
        counter++;
        if (counter == 9) 
        {
            //do your task for 3 months
            //since counter increments 9 times at interval of 10 days
            //so 9*10=90 days i.e. nearly equal to 3 months
            counter = 0;
        }
    }

答案 2 :(得分:1)

我会使用Quartz.net - 开源企业作业调度程序 - http://quartznet.sourceforge.net/

您可以使用CRON语法 - http://quartznet.sourceforge.net/tutorial/lesson_6.html &安培;你可以很容易地将相关的工作状态存储到数据库中(只是让机器死掉,它会发生) - http://quartznet.sourceforge.net/tutorial/lesson_9.html,以及更多的实用性。

我在几个生产系统上使用了Quartz.net,据我所知,这些流程今天仍在运行:)

为什么不使用System.Timer - http://quartznet.sourceforge.net/faq.html#whynottimer

祝你好运!

答案 3 :(得分:1)

有同样的问题,除了它是一个通用的系统,可以用完全可调节的间隔做同样的事情,从几毫秒到几个月。那么,应该到。事实证明,由于这个原因,它确实在大于24.8天的间隔内搞砸了。

就我而言,&#34;重新思考方法&#34;这是不可能的,因为它只是一个更大的Windows服务系统的一个小问题。

解决方案相当简单,但请注意我有一个辅助系统,为我提供下一次执行DateTime;计时器的工作只是为了与实际执行任务相匹配,所以它只需从中减去DateTime.Now来计算间隔。

在这种情况下,我需要做的就是在我的计时器对象旁边保留一个溢出布尔值,并在检查Int32.MaxValue的间隔时设置:

private Timer _timer;
private Boolean _overflow;

private void QueueNextTime(DateTime thisTime)
{
    TimeSpan interval = this.GetNextRunTime(thisTime) - DateTime.Now;
    Int64 intervalInt = (Int64)((interval.TotalMilliseconds <= 0) ? 1 : interval.TotalMilliseconds);
    // If interval greater than Int32.MaxValue, set the boolean to skip the next run. The interval will be topped at Int32.MaxValue.
    // The TimerElapsed function will call this function again anyway, so no need to store any information on how much is left.
    // It'll just repeat until the overflow status is 'false'.
    this._overflow = intervalInt > Int32.MaxValue;
    this._timer.Interval = Math.Min(intervalInt, Int32.MaxValue);
    this._timer.Start();
}

// The function linked to _timer.Elapsed
private void TimerElapsed(object sender, ElapsedEventArgs e)
{
    this._timer.Stop();
    if (this._overflow)
    {
        QueueNextTime(e.SignalTime);
        return;
    }

    // Execute tasks
    // ...
    // ...

    // schedule next execution
    QueueNextTime(e.SignalTime);
}

当然,这是简化的;真正的系统有try / catch和一个系统来中止外部停止命令。但这就是它的主旨。

答案 4 :(得分:0)

我会创建一个exe应用程序并作为计划的Windows任务的一部分运行。