后台工作程序如何检测午夜时刻?

7

我希望为WinForm创建一个后台工作者,当午夜到来时触发代码。

我有一个想法,但我相信这不是最好的方法。

while(1==1)
{
//if Datetime.Now == midnight, execute code
//sleep(1second)
}

快速提醒... 如果你想创建一个“无限”的循环(当然可以有break),你可以使用while(true) { }1==1本身就会被评估为true - Nelson Rothermel
那是被广泛接受的标准吗?我一直对使用1==1是否合适不太确定。 - sooprise
6个回答

23

使用 System.Timers.Timer,在应用程序启动时计算 DateTime.NowDateTime.Today.AddDays(0)之间的差异。然后将定时间隔设置为该差值。

我最近也做了类似的事情:

public static class DayChangedNotifier
{
    private static Timer timer;

    static DayChangedNotifier()
    {
        timer = new Timer(GetSleepTime());
        timer.Elapsed += (o, e) =>
            {
                OnDayChanged(DateTime.Now.DayOfWeek);
                timer.Interval = this.GetSleepTime();
            };
        timer.Start();

        SystemEvents.TimeChanged += new EventHandler(SystemEvents_TimeChanged);
    }

    private static void SystemEvents_TimeChanged(object sender, EventArgs e)
    {
        timer.Interval = GetSleepTime();
    }

    private static double GetSleepTime()
    {
        var midnightTonight = DateTime.Today.AddDays(1);
        var differenceInMilliseconds = (midnightTonight - DateTime.Now).TotalMilliseconds;
        return differenceInMilliseconds;
    }

    private static void OnDayChanged(DayOfWeek day)
    {
        var handler = DayChanged;
        if (handler != null)
        {
            handler(null, new DayChangedEventArgs(day));
        }
    }

    public static event EventHandler<DayChangedEventArgs> DayChanged;
}

与(AND):

public class DayChangedEventArgs : EventArgs
{
    public DayChangedEventArgs(DayOfWeek day)
    {
        this.DayOfWeek = day;
    }

    public DayOfWeek DayOfWeek { get; private set; }
}

用法:DayChangedNotified.DayChanged += ....


@Soo:不会的。计时器不会阻塞(停止你的其余代码 - http://en.wikipedia.org/wiki/Blocking_%28computing%29)。一旦时间间隔到期,它将调用“Tick”事件。此时,如果您希望停止计时器,则可以停止计时器,计算下一个时间间隔并再次启动它。 - Nelson Rothermel
不,System.Timers 在另一个线程上触发,所以你需要小心,在计时器的事件处理程序中调用 Invoke。 - BFree
或者使用 System.Windows.Forms.Timer 替代,它会在窗体线程中触发 Tick() 事件。 - Jürgen Steinblock

7

相反,您可以使用计时器,并将计时器的间隔设置为现在()和午夜之间的时间。


1
我不知道为什么投票解决方案会得到支持,因为微软多年前通过添加Windows服务来处理定时问题。只需创建一个计划任务来运行exe文件,无需额外的开销。

0
不要使用轮询。相反,设置一个定时器任务,将其设置为在午夜触发,并添加一个事件进行处理。
 TimeSpan timeBetween = DateTime.Today.AddDays(1) - DateTime.Now;

 System.Timers.Timer t = new System.Timers.Timer();
 t.Elapsed += new System.Timers.ElapsedEventHandler(t_Elapsed);
 t.Interval = 1000 * timeBetween.Seconds;
 t.Start();

那么在t_Elapsed被触发后,timeBetween会被重新计算吗? - sooprise
你需要在 t_Elapsed 方法中重新计算时间间隔。 - cortijon

0

你可以使用Quartz来进行调度。也许在这种情况下有点大材小用,但这是我所知道的唯一一个调度作业框架,而且表现非常出色。


0
我有点困惑为什么你需要一个WinForm,它会在午夜运行吗?如果你只需要运行某种进程,请使用Windows计划程序在午夜运行它。(在XP上,但我相信Win服务器应该类似)控制面板->计划任务->添加计划任务->填写向导。这样可以节省很多编码工作。

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接