我最初尝试使用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);
答案 0 :(得分:3)
如果我正确阅读您的评论,您正在从非UI线程调用Dispatcher.CurrentDispatcher
。这不是如何使用的。
正如Dispatcher.CurrentDispatcher的文档所说:
获取当前正在执行 的线程的Dispatcher ,并创建一个新的Dispatcher,如果还没有 与该线程 < /强>
要获取有效的调度程序实例,您需要从UI线程调用Dispatcher.CurrentDispatcher
。
此外,因为文档说它会自动创建一个调度程序,如果当前线程不存在,这就是解释静默失败的原因。您正在获取调度程序实例,但它并未以任何方式与UI线程关联,因此它实际上并未向UI线程分派任何内容。
(删除它,因为在我的测试中,即使我不应该,我也会变为空,所以它看起来并不多。其余的信息是准确的)
文档还添加:
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
。