如何在WinForm线程上获取WinForm同步上下文或计划

时间:2011-09-14 14:25:26

标签: winforms .net-4.0 system.reactive synchronizationcontext

我有一个winform应用程序,以及一个可观察的设置:

Form form = new Form();
Label lb = new Label();
form.Controls.Add(lb);

Observable.Interval(TimeSpan.FromSeconds(1))
          .Subscribe(l => lb.Text = l.ToString());

Application.Run(form);

这不起作用,因为l => lb.Text = l.ToString()不会在创建表单的主线程上运行,但我无法弄清楚如何使它在这个线程上运行。我假设,我应该IObservable.SubscribeOn使用ISchedulerSynchronizationContext,但我不知道如何获取主线程的同步上下文,以及唯一的调度程序I可以找到Scheduler的静态属性,例如Scheduler.CurrentThreadImmediateNewThreadTaskPoolThreadPool,其中没有一个有效。

我的Rx版本是1.0.10621。

1 个答案:

答案 0 :(得分:23)

在我发布问题之后,我找到了解决方案:

Form form = new Form();
Label lb = new Label();
form.Controls.Add(lb);

Observable.Interval(TimeSpan.FromSeconds(2))
          .ObserveOn(SynchronizationContext.Current)
          .Subscribe(l => lb.Text = l.ToString());

Application.Run(form);

This链接很有帮助。两个注意事项:

  • 并非所有线程都有同步上下文,但在线程上创建的第一个表单将为该线程设置同步上下文,因此UI线程始终有一个。
  • 正确的方法是ObserveOn,而不是SubscribeOn。目前,我对此知之甚少,知道原因,所以评论中的任何链接都会受到赞赏。

编辑:感谢this链接的第一部分,我现在更了解ObserveOnSubscribeOn之间的区别。简而言之:

  • 观察同步上下文的observable将从该上下文调用IObserver方法(OnNext和朋友)。在我的例子中,我在main / UI线程上观察,因此我没有得到跨线程异常
  • SubscribeOn稍微复杂一些,所以这里有一个例子:Concat采用了许多可观察对象并将它们组合成一个长的可观察对象。一旦observable调用OnCompleted,组合的observable将处理该订阅,并订阅下一个observable。这一切都发生在调用OnCompleted的线程上,但是订阅可由Observable.FromEvent创建的observable可能存在一些问题,例如SubscribeOn。如果从除UI线程之外的其他线程添加事件处理程序,Silverlight将抛出,如果从多个不同的线程添加事件处理程序,则将抛出WinForms和WPF。 {{1}}将允许您控制observable连接到基础事件的线程。