Only raise an event if the previous one was completed

后端 未结 4 780
误落风尘
误落风尘 2021-01-16 11:25

I\'m using a System.Timers.Timer in my application. Every second I run a function which does some job. The thing is, this function can block for some little tim

4条回答
  •  时光说笑
    2021-01-16 12:09

    Easiest way I know of to do this kind of thing:

    internal static volatile bool isRunning;
    
    public static void Main()
        {
            Thread TT = new Thread(new ThreadStart(delegate()
                {
                    System.Timers.Timer oTimer = new System.Timers.Timer();
                    oTimer.Elapsed += new ElapsedEventHandler(Handler);
    
                    oTimer.Interval = 1000;
                    oTimer.Enabled = true;                 
                }));
    
            TT.Start();
        }
    
        private void Handler(object oSource,
            ElapsedEventArgs oElapsedEventArgs)
        {
            if(isRunning) return;
    
            isRunning = true;
    
            try
            {
               Console.WriteLine("foo");
               Thread.Sleep(500);         //simulate some work
               Console.WriteLine("bar");
            }
            finally { isRunning = false; }            
        }
    

    The handler still runs, but the very first thing it does is make sure that another handler isn't running, and if one is, it stops immediately.

    For timers executing handlers more quickly (like 3-4 times a second), this has the possibility to race; two threads could proceed past the guard clause before one of them sets the bit. You can avoid this with a couple of lock statements, similar to a Mutex or Monitor:

        static object syncObj = new object();
    
        private void Handler(object oSource,
            ElapsedEventArgs oElapsedEventArgs)
        {
            lock(syncObj)
            {
               if(isRunning) return;            
               isRunning = true;
            }
    
            try
            {
               Console.WriteLine("foo");
               Thread.Sleep(500);         //simulate some work
               Console.WriteLine("bar");
            }
            finally { lock(syncObj) { isRunning = false; } }            
        }
    

    This will ensure that only one thread can ever be examining or modifying isRunning, and as isRunning is marked volatile, the CLR won't cache its value as part of each thread's state for performance; each thread has to look at exactly the same memory location to examine or change the value.

提交回复
热议问题