Windows服务中使用的最佳计时器


108

我需要创建一些Windows服务,每隔N个时间段执行一次。
问题是:
我应该使用哪个计时器控件:System.Timers.Timer还是System.Threading.Timer一个?它会影响某些东西吗?

我之所以问是因为我听到了很多证据证明System.Timers.TimerWindows服务的工作不正确。
谢谢。

Answers:


118

双方System.Timers.TimerSystem.Threading.Timer会为服务工作。

您要避免使用的计时器是System.Web.UI.TimerSystem.Windows.Forms.Timer,分别用于ASP应用程序和WinForms。使用这些将导致服务加载额外的程序集,而对于您正在构建的应用程序类型,这实际上并不需要。

使用System.Timers.Timer类似下面的示例(此外,请确保您使用类级变量来防止垃圾收集,如Tim Robinson的回答所述):

using System;
using System.Timers;

public class Timer1
{
    private static System.Timers.Timer aTimer;

    public static void Main()
    {
        // Normally, the timer is declared at the class level,
        // so that it stays in scope as long as it is needed.
        // If the timer is declared in a long-running method,  
        // KeepAlive must be used to prevent the JIT compiler 
        // from allowing aggressive garbage collection to occur 
        // before the method ends. (See end of method.)
        //System.Timers.Timer aTimer;

        // Create a timer with a ten second interval.
        aTimer = new System.Timers.Timer(10000);

        // Hook up the Elapsed event for the timer.
        aTimer.Elapsed += new ElapsedEventHandler(OnTimedEvent);

        // Set the Interval to 2 seconds (2000 milliseconds).
        aTimer.Interval = 2000;
        aTimer.Enabled = true;

        Console.WriteLine("Press the Enter key to exit the program.");
        Console.ReadLine();

        // If the timer is declared in a long-running method, use
        // KeepAlive to prevent garbage collection from occurring
        // before the method ends.
        //GC.KeepAlive(aTimer);
    }

    // Specify what you want to happen when the Elapsed event is 
    // raised.
    private static void OnTimedEvent(object source, ElapsedEventArgs e)
    {
        Console.WriteLine("The Elapsed event was raised at {0}", e.SignalTime);
    }
}

/* This code example produces output similar to the following:

Press the Enter key to exit the program.
The Elapsed event was raised at 5/20/2007 8:42:27 PM
The Elapsed event was raised at 5/20/2007 8:42:29 PM
The Elapsed event was raised at 5/20/2007 8:42:31 PM
...
 */

如果选择System.Threading.Timer,则可以使用以下方法:

using System;
using System.Threading;

class TimerExample
{
    static void Main()
    {
        AutoResetEvent autoEvent     = new AutoResetEvent(false);
        StatusChecker  statusChecker = new StatusChecker(10);

        // Create the delegate that invokes methods for the timer.
        TimerCallback timerDelegate = 
            new TimerCallback(statusChecker.CheckStatus);

        // Create a timer that signals the delegate to invoke 
        // CheckStatus after one second, and every 1/4 second 
        // thereafter.
        Console.WriteLine("{0} Creating timer.\n", 
            DateTime.Now.ToString("h:mm:ss.fff"));
        Timer stateTimer = 
                new Timer(timerDelegate, autoEvent, 1000, 250);

        // When autoEvent signals, change the period to every 
        // 1/2 second.
        autoEvent.WaitOne(5000, false);
        stateTimer.Change(0, 500);
        Console.WriteLine("\nChanging period.\n");

        // When autoEvent signals the second time, dispose of 
        // the timer.
        autoEvent.WaitOne(5000, false);
        stateTimer.Dispose();
        Console.WriteLine("\nDestroying timer.");
    }
}

class StatusChecker
{
    int invokeCount, maxCount;

    public StatusChecker(int count)
    {
        invokeCount  = 0;
        maxCount = count;
    }

    // This method is called by the timer delegate.
    public void CheckStatus(Object stateInfo)
    {
        AutoResetEvent autoEvent = (AutoResetEvent)stateInfo;
        Console.WriteLine("{0} Checking status {1,2}.", 
            DateTime.Now.ToString("h:mm:ss.fff"), 
            (++invokeCount).ToString());

        if(invokeCount == maxCount)
        {
            // Reset the counter and signal Main.
            invokeCount  = 0;
            autoEvent.Set();
        }
    }
}

这两个示例均来自MSDN页面。


1
您为什么建议:GC.KeepAlive(aTimer);, aTimer是正确的实例变量,所以例如,如果它是形式的实例变量,那么只要存在形式,就总会引用它,不是吗?
giorgim 2015年

37

请勿为此使用服务。创建一个普通应用程序并创建一个计划任务以运行它。

这是通常的最佳做法。 乔恩·加洛韦(Jon Galloway)同意我的观点。或相反。 无论哪种方式,事实都是创建Windows服务以执行在计时器外运行的间歇性任务不是最佳实践。

“如果要编写运行计时器的Windows服务,则应重新评估解决方案。”

– Jon Galloway,ASP.NET MVC社区计划经理,作者,兼职超级英雄


26
如果您的服务打算全天运行,那么也许服务有意义,而不是计划任务。或者,拥有一项服务可能会简化不如应用程序团队那么精明的基础结构组的管理和日志记录。但是,质疑需要服务的假设是完全有效的,并且这些负面评价是不值得的。两者都+1。
sfuqua

2
@废话。计划任务是操作系统的核心部分。请把您的FUD留给自己。

4
@MR:我不是传教士,我是现实主义者。现实告诉我,计划任务并不是“极端的越野车”。实际上,取决于您是谁提出支持该声明的人。否则,您正在做的就是散布恐惧,不确定性和怀疑。

13
我不想涉足这里的泥潭,但我必须为MR辩护。我的公司中运行着几个重要的应用程序,它们是Windows控制台应用程序。我已经使用Windows Task Scheduler来运行它们。现在,至少有5次出现了一个问题,调度程序服务以某种方式“变得混乱”。任务未执行,有些处于奇怪状态。唯一的解决方案是重新启动服务器或停止并启动Scheduler服务。没有管理员权限,在生产环境中我不能做些什么。就是我的$ 0.02。
SpaceCowboy74 2013年

6
实际上,这是在几台服务器上发生的(到目前为止,有3台)。虽然不会说这是常态。只是说有时候实现自己的做某事的方法还不错。
SpaceCowboy74

7

任一个都可以正常工作。实际上,System.Threading.Timer在内部使用System.Timers.Timer。

话虽如此,很容易滥用System.Timers.Timer。如果您没有将Timer对象存储在某个地方的变量中,则很可能会被垃圾回收。如果发生这种情况,您的计时器将不再触发。调用Dispose方法停止计时器,或使用System.Threading.Timer类,该类稍微好一些。

到目前为止,您看到了什么问题?


让我想知道为什么Windows Phone应用只能访问System.Threading.Timer。
Stonetip

Windows Phone可能具有框架的简化版本,可能不需要具有所有额外代码即可使用这两种方法,因此不包括在内。我认为,尼克(Nick)的回答为Windows手机无法访问(System.Timers.Timer因为它无法处理抛出的异常)提供了更好的理由。
玛拉基

2

我同意先前的评论,最好是考虑使用其他方法。我的建议是编写一个控制台应用程序并使用Windows Scheduler:

这将:

  • 减少复制调度程序行为的管道代码
  • 从应用程序代码中提取所有调度逻辑,从而在调度行为(例如仅在周末运行)方面提供更大的灵活性
  • 利用命令行参数获取参数,而无需在配置文件等中设置配置值
  • 在开发过程中进行调试/测试要容易得多
  • 通过直接调用控制台应用程序来允许支持用户执行(例如,在支持情况下很有用)

3
但这需要已登录的用户吗?因此,如果它将在服务器上全天候运行24/7,服务可能会更好。
JP Hellemons,2011年

1

如前所述都System.Threading.TimerSystem.Timers.Timer正常工作。两者之间的最大区别System.Threading.Timer是另一个包装器。

System.Threading.Timer将拥有更多的异常处理能力,同时 System.Timers.Timer吞噬所有异常。

过去这给我带来了很大的问题,因此我将始终使用“ System.Threading.Timer”,并且仍然可以很好地处理您的异常。


0

我知道这个线程有些旧,但是对于我遇到的特定情况它很有用,我认为值得一提的是,还有另一个原因System.Threading.Timer可能是一个好的方法。如果您必须定期执行可能需要很长时间的作业,并且想要确保整个作业之间都使用了整个等待时间,或者如果您不希望作业在之前的作业完成之前再次运行,则需要这项工作所花费的时间超过了计时器时间。您可以使用以下内容:

using System;
using System.ServiceProcess;
using System.Threading;

    public partial class TimerExampleService : ServiceBase
    {
        private AutoResetEvent AutoEventInstance { get; set; }
        private StatusChecker StatusCheckerInstance { get; set; }
        private Timer StateTimer { get; set; }
        public int TimerInterval { get; set; }

        public CaseIndexingService()
        {
            InitializeComponent();
            TimerInterval = 300000;
        }

        protected override void OnStart(string[] args)
        {
            AutoEventInstance = new AutoResetEvent(false);
            StatusCheckerInstance = new StatusChecker();

            // Create the delegate that invokes methods for the timer.
            TimerCallback timerDelegate =
                new TimerCallback(StatusCheckerInstance.CheckStatus);

            // Create a timer that signals the delegate to invoke 
            // 1.CheckStatus immediately, 
            // 2.Wait until the job is finished,
            // 3.then wait 5 minutes before executing again. 
            // 4.Repeat from point 2.
            Console.WriteLine("{0} Creating timer.\n",
                DateTime.Now.ToString("h:mm:ss.fff"));
            //Start Immediately but don't run again.
            StateTimer = new Timer(timerDelegate, AutoEventInstance, 0, Timeout.Infinite);
            while (StateTimer != null)
            {
                //Wait until the job is done
                AutoEventInstance.WaitOne();
                //Wait for 5 minutes before starting the job again.
                StateTimer.Change(TimerInterval, Timeout.Infinite);
            }
            //If the Job somehow takes longer than 5 minutes to complete then it wont matter because we will always wait another 5 minutes before running again.
        }

        protected override void OnStop()
        {
            StateTimer.Dispose();
        }
    }

    class StatusChecker
        {

            public StatusChecker()
            {
            }

            // This method is called by the timer delegate.
            public void CheckStatus(Object stateInfo)
            {
                AutoResetEvent autoEvent = (AutoResetEvent)stateInfo;
                Console.WriteLine("{0} Start Checking status.",
                    DateTime.Now.ToString("h:mm:ss.fff"));
                //This job takes time to run. For example purposes, I put a delay in here.
                int milliseconds = 5000;
                Thread.Sleep(milliseconds);
                //Job is now done running and the timer can now be reset to wait for the next interval
                Console.WriteLine("{0} Done Checking status.",
                    DateTime.Now.ToString("h:mm:ss.fff"));
                autoEvent.Set();
            }
        }
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.