DispatcherTimer vs a regular Timer in WPF app for a task scheduler

46,483

Solution 1

DispatcherTimer is the regular timer. It fires its Tick event on the UI thread, you can do anything you want with the UI. System.Timers.Timer is an asynchronous timer, its Elapsed event runs on a thread pool thread. You have to be very careful in your event handler, you are not allowed to touch any UI component or data-bound variables. And you'll need to use the lock statement where ever you access class members that are also used on the UI thread.

In the linked answer, the Timer class was more appropriate because the OP was trying to run code asynchronously on purpose.

Solution 2

Regular Timer's Tick event is actually fired on the thread where the Timer was created, so in the tick event, in order to access anything with the UI, you will have to go through dispatcher.begininvoke as mentioned below.

RegularTimer_Tick(object sender, EventArgs e)
{
    txtBox1.Text = "count" + i.ToString(); 
    // error can not access
    // txtBox1.Text property outside dispatcher thread...

    // instead you have to write...
    Dispatcher.BeginInvoke( (Action)delegate(){
       txtBox1.Text = "count " + i.ToString();
    });
}

In case of Dispatcher Timer, you can access UI elements without doing begin invoke or invoke as follow...

DispatcherTimer_Tick(object sender, EventArgs e)
{
    txtBox1.Text = "Count " + i.ToString();
    // no error here..
}

DispatcherTimer just provides convenience over Regular Timer to access UI objects easily.

Solution 3

With .NET 4.5 you can also create an async delegate for your timer if you need to use the new .NET 4.5 await functionality.

        var timer = new DispatcherTimer();
        timer.Interval = TimeSpan.FromSeconds(20);
        timer.Tick += new EventHandler(async (object s, EventArgs a) =>
        {
            Message = "Updating...";
            await UpdateSystemStatus(false);  
            Message = "Last updated " + DateTime.Now;              
        });
        timer.Start();
Share:
46,483
rem
Author by

rem

Updated on July 09, 2022

Comments

  • rem
    rem almost 2 years

    Please, explain the difference between "DispatcherTimer" and "a regular Timer" that @Kent Boogaart meant for using in a multithreading WPF app as a task sheduler in this topic:

    Advice needed for multi-threading strategy for WPF application

    in the commentaries to one of the post (quote):

    -If all the DispatcherTimer does is kick off another thread, what is the point of using the DispatcherTimer? ....those threads don't need to be started on the UI thread. You could just use a regular Timer and avoid interrupting the UI altogether

    What is "a regular timer" that is meant? How they ("DispatcherTimer" and "a regular Timer") differ regarding their impact on UI?

    (Until reading this post I thought about DispatcherTimer as a natural way of using timers in WPF. What is the cases when this is not true?)

  • aruno
    aruno over 11 years
    note that 'touching' includes setting MVVM viewmodel properties so when using a DispatcherTimer you're fine, but with a Timer you can't even do that
  • aruno
    aruno over 11 years
    this may also be of interest (my own question seeking an async friendly DispatcherTimer) stackoverflow.com/questions/12442622/…
  • Martin Ch
    Martin Ch about 8 years
    is dispatcher Timer using Invoke or BeginInvoke?
  • Akash Kava
    Akash Kava about 8 years
    Based on the source here, DispatcherTimer is using BeginInvoke, reason is, dispatcher queue has priority based invoke mechanism. referencesource.microsoft.com/#WindowsBase/Base/System/Windo‌​ws/…
  • Steven Rands
    Steven Rands almost 4 years
    @Simon_Weaver If it's a basic string/number/Boolean property then changing the value in the timer's callback will work just fine, even if the callback doesn't run on the UI thread. It is true that you can encounter problems with something like an ObservableCollection property though.