我的服务会多次运行日常任务。这是我的服务代码:
private DateTime m_scheduleTime;
private List<Task> m_tasksList; // Task includes id and time ("23:00")
protected override void OnStart(string[] args) {
SetSchedule();
}
private void SetSchedule()
{
Timer[] timers = new Timer[m_tasksList.Count];
int iHours = 0; int iMinutes = 0;
for (int i = 0; i < timers.Length; i++)
{
iHours = Int16.Parse(m_tasksList[i].Time.Split(':')[0]);
iMinutes = Int16.Parse(m_tasksList[i].Time.Split(':')[1]);
m_scheduleTime = DateTime.Today.AddDays(1).AddHours(iHours).AddMinutes(iMinutes);
timers[i] = new Timer();
timers[i].Enabled = true;
timers[i].Interval = m_scheduleTime.Subtract(DateTime.Now).TotalSeconds * 1000;
timers[i].Elapsed += new System.Timers.ElapsedEventHandler(Timer_Elapsed);
}
}
protected void Timer_Elapsed(object sender, System.Timers.ElapsedEventArgs e)
{
DoWork(int taskId);
}
上面的代码没有经过测试,我是从相关的代码示例中构建的。所以我的问题是我不知道我是否在正确的轨道上,如果我是对的,我不知道哪个计时器触发_Elapsed事件,以便我可以传递相应的参数。任何更好的解决方案将不胜感激!
答案 0 :(得分:2)
使用单个System.Threading.Timer
可能会获得更好的结果,该OlaMagic.DashboardIndexRoute = Ember.Route.extend({
model: function(params) {
return this.store.findAll('number');
}
});
会以最低分辨率(例如每60秒)定期到期。
当您的服务启动时,计算所有任务的下一步预定运行时间,然后将它们插入优先级队列。优先级队列应按计划的运行时间排序。
每次计时器到期时,评估队列头部任务的预定运行时间。如果已达到任务的运行时间,则从队列中弹出它并将其推送到队列,工作线程将执行该队列。重复此过程,直到您在队列中用完任务或找到尚未准备好运行的任务。
任务完成后,计算下一个计划的运行时间并将其插入计划的优先级队列。
如果您的服务纯粹是为了运行这些计划任务,我建议您考虑使用Windows任务计划程序。老实说,即使您的服务提供其他功能,使用Windows任务计划程序运行定期任务仍然是比尝试滚动自己的计划程序更好的选择。
您可以在此处http://taskscheduler.codeplex.com/找到Windows任务计划程序的托管包装程序。我不知道这个包装器,我发现它在这个问题中被引用:C# API for Task Scheduler 2.0。