【问题标题】:Why did dispatcher BeginInvoke fail where Control BeginInvoke succeed in C# Windows Forms app?为什么调度程序 BeginInvoke 在 C# Windows Forms 应用程序中 Control BeginInvoke 成功时失败?
【发布时间】:2015-07-30 00:14:38
【问题描述】:

我最初尝试使用 Dispatcher 类 BeginInvoke 方法在我的 C# Windows 窗体应用程序的主 UI 线程上显示一个消息框。当我使用该方法时,消息框没有出现。我在传递给 BeginInvoke() 的委托主体内设置了一个断点,它从未命中。我尝试同时使用 Action 委托和 MethodInvoker 委托。两种情况都没有运气。

当我使用属于 Form 对象的 BeginInvoke 方法时,它运行良好。为什么 Dispatch 版本会静默失败(没有异常或错误消息)?以下是两个不同的版本。

Dispatcher dispatcher = Dispatcher.CurrentDispatcher;

// THIS FAILED. CONTEXT: Executing on worker thread.
MethodInvoker theMethod = new MethodInvoker(delegate()
{
    string msg = "Show this  message on the main UI thread.";
    MessageBox.Show(msg, "Message");
});

dispatcher.BeginInvoke(theMethod);

this.BeginInvoke(theMethod);

// ---------------------------------------------------

// THIS WORKED. CONTEXT: Executing on worker thread.
MethodInvoker theMethod = new MethodInvoker(delegate()
{
    string msg = "Show this  message on the main UI thread.";
    MessageBox.Show(msg, "Message");
});

// "this" is a Form object.
this.BeginInvoke(theMethod);

【问题讨论】:

  • 似乎此页面包含如何将Dispatcher 与winforms 一起使用的示例:stackoverflow.com/questions/303116/…
  • 我尝试了你的 dispatcher.BeginInvoke(...); 代码,它在 .NET4 中运行良好。
  • @Loathing 感谢您的测试。我(不止)在测试期间进行了三次检查,我可以向你保证,对我来说它绝对没有用。所以现在我们显然遇到了一个非常奇怪的情况。
  • @RobertOschler:这里已经有一个很好的答案,但作为旁注:您不应该使用任何BeginInvoke 方法。要报告进度,请使用IProgress<T>。要使用结果更新 UI,请使用 await。要从异步流动态更新 UI,请使用 Rx 的ObserveOn(SynchronizationContext)。这些较新的方法将您的逻辑与您的 UI 分离,并且可以进行单元测试。
  • @StephenCleary 谢谢斯蒂芬。

标签: c# winforms dispatcher begininvoke


【解决方案1】:

如果我正确读取了您的 cmets,则您正在从非 UI 线程调用 Dispatcher.CurrentDispatcher。这不是它的用途。

正如Dispatcher.CurrentDispatcher 的文档所说:

为当前正在执行的线程获取 Dispatcher,如果尚未与线程关联

要获得有效的调度程序实例,您需要从 UI 线程调用 Dispatcher.CurrentDispatcher

此外,因为文档说如果当前线程不存在调度程序,它将自动创建一个调度程序,这就是静默失败的原因。您正在获取一个调度程序实例,但它没有以任何方式与 UI 线程关联,因此它实际上并没有向 UI 线程调度任何内容。

(删除这个,因为在我的测试中,即使我不应该得到 null,所以它似乎并不能证明太多。但其余信息是准确的) 文档中还添加了:

FromThread 方法不是这种情况。如果没有与指定线程关联的调度程序,FromThread 将返回 null

因此,要确认您确实获得了自动创建的(无效)调度程序,请尝试从 Dispatcher.FromThread 获取调度程序。我猜你会得到null

如果您想从工作线程调用dispatcher.BeginInvoke 来强制执行UI 线程上的方法,您需要从UI 线程调用Dispatcher.CurrentDispatcher 并将其保存到变量中。然后,您可以将该调度程序引用变量传递给工作线程,并在其上调用 BeginInvoke

// capture and save dispatcher from UI thread
Dispatcher dispatcher = Dispatcher.CurrentDispatcher;

// then you can do this from your worker thread:
dispatcher.BeginInvoke(theMethod);

或者,使用this.BeginInvoke,就像你已经在做的那样。

或者更好的是,您可以尝试将任务与新的 async-await 关键字结合使用来完成此类事情。

编辑

为了完整起见,我应该解释一下为什么Control.BeginInvoke 可以正常工作。

正如Control.BeginInvoke 的文档所说:

在创建控件的基础句柄的线程上异步执行指定的委托。

后来还加了:

您可以从任何线程调用此方法。

关键是,当您调用Control.BeginInvoke 时,它不会使用当前线程来确定如何执行委托。它会记住控件是在哪个线程(UI 线程)上创建的,并确保在该线程上执行委托。

因此,只要您的控件是在 UI 线程上创建的(应该如此),BeginInvoke 就可以在任何线程上工作。这其实和Dispatcher很像,只要你先从UI线程获取Dispatcher实例,然后你也可以从任意线程调用Dispatcher.BeginInvoke

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2013-09-24
    • 2011-04-16
    • 1970-01-01
    相关资源
    最近更新 更多