Consider the following code for a WPF ViewModel:
protected void Init()
{
Debug.WriteLine(string.Format("ChangeManager init on thread={0}", Thread.CurrentThread.ManagedThreadId));
var uiTaskScheduler = TaskScheduler.FromCurrentSynchronizationContext();
this.modelChanged = (o, args) => Task.Factory.StartNew(() =>
{
Debug.WriteLine(string.Format("ModelChanged on thread={0}", Thread.CurrentThread.ManagedThreadId));
this.ModelChanged(o, args);
},
CancellationToken.None,
TaskCreationOptions.None,
uiTaskScheduler);
}
... where modelChanged is an event handler for responding to changes in an object model. This code executes on the UI thread and is designed in hopes of having events handled on the UI thread regardless of which thread they are fired from.
However, when this gets run the output is something similar to:
ChangeManager init on thread=1
ModelChanged on thread=3
ModelChanged on thread=3
ModelChanged on thread=7
ModelChanged on thread=9
My expectation would be that thread 1 would be where all handling would take place. Even when I try using the SynchronizationContext directly like this:
protected void Init()
{
Debug.WriteLine(string.Format("ChangeManager init on thread={0}", Thread.CurrentThread.ManagedThreadId));
this.uiContext = SynchronizationContext.Current;
modelChanged = (o, args) => uiContext.Post((ignore) => {
Debug.WriteLine(string.Format("ModelChanged on thread={0}", Thread.CurrentThread.ManagedThreadId));
this.ModelChanged(o, args);
}
, null);
}
... I see the same thing.
Is there something wrong with my thinking or approach? How do I get the events to be handled on the init thread?
Thanks in advance!
Interesting, your code works for me. Maybe you left out parts of the code that can explain the problem. Can you post a more complete reproduction of the problem? And specifically, show what you are doing with the
modelChanged
member other than assigning the lambda to it.What I did was, create an empty WPF application and run your Init method from the main window's constructor.
Then I started background threads that called the
modelChanged
delegate directly.What I saw was that the line "ModelChanged on thread..." always printed the correct thread, the one which called
Init
.If it's any help, here's what I did to try to reproduce it, you can look at it and maybe post about what you're doing differently: