Invoke()和BeginInvoke()有什么区别?

只是想知道BeginInvoke()Invoke()之间的区别是什么?

主要是它们各自的用途。

编辑:创建线程对象和调用调用线程对象与在委托上调用BeginInvoke()有什么区别?还是说它们是一样的?

212088 次浏览

你是指Delegate.Invoke/BeginInvoke还是Control.Invoke/BeginInvoke?

  • Delegate.Invoke:同步执行,在同一个线程上。
  • Delegate.BeginInvoke:在threadpool线程上异步执行。
  • Control.Invoke:在UI线程上执行,但是调用线程在继续之前等待完成。
  • Control.BeginInvoke:在UI线程上执行,并且调用线程不等待完成。

Tim的回答提到了你可能想要使用BeginInvoke的时候——尽管我怀疑它主要是针对Delegate.BeginInvoke的。

对于Windows窗体应用程序,我建议你应该通常使用BeginInvoke。这样你就不需要担心死锁,但是你需要明白UI在你下次看到它的时候可能还没有更新!特别是,您不应该修改UI线程可能用于显示目的的数据。例如,如果你有一个具有FirstNameLastName属性的Person,并且你做了:

person.FirstName = "Kevin"; // person is a shared reference
person.LastName = "Spacey";
control.BeginInvoke(UpdateName);
person.FirstName = "Keyser";
person.LastName = "Soze";

然后UI很可能会显示“Keyser Spacey”。(它有很小的可能会显示“凯文·苏斯”,但这只能通过奇怪的内存模型来实现。)

然而,除非你有这样的问题,Control.BeginInvoke更容易得到正确的,并将避免你的后台线程不得不等待没有好的理由。注意,Windows窗体团队已经保证你可以以一种“触发并忘记”的方式使用Control.BeginInvoke——即不调用EndInvoke。一般来说,异步调用不是这样的:通常每个BeginXXX都应该有一个相应的EndXXX调用,通常是在回调中。

delegate . begininvoke()异步地对委托的调用进行排队,并立即返回控制。当使用Delegate.BeginInvoke()时,你应该在回调方法中调用Delegate.EndInvoke()来获得结果。

delegate . invoke()在同一个线程中同步调用委托。

MSDN文章

根据Jon Skeet的回答,有时候你想要调用委托,并在当前线程继续之前等待它的执行完成。在这些情况下,Invoke调用就是您想要的。

在多线程应用程序中,您可能不希望线程等待委托完成执行,特别是当委托执行I/O时(这可能会使委托和线程阻塞)。

在这些情况下,BeginInvoke会很有用。通过调用它,你告诉委托开始,然后你的线程可以自由地与委托并行做其他事情。

使用BeginInvoke会增加代码的复杂性,但有时性能的提高是值得的。

Control.Invoke()Control.BeginInvoke()的区别是,

  • BeginInvoke()将在GUI线程上调度异步操作。当异步操作被调度时,代码将继续执行。一段时间后(您不知道确切的时间)将执行异步操作
  • Invoke()将执行你的异步操作(在GUI线程上)并等待直到你的操作完成。

一个合乎逻辑的结论是,传递给Invoke()的委托可以有外参数或返回值,而传递给BeginInvoke()的委托则不能(必须使用EndInvoke检索结果)。

举个简单的例子,看看它们的差异会产生什么影响

new Thread(foo).Start();


private void foo()
{
this.Dispatcher.BeginInvoke(DispatcherPriority.Normal,
(ThreadStart)delegate()
{
myTextBox.Text = "bing";
Thread.Sleep(TimeSpan.FromSeconds(3));
});
MessageBox.Show("done");
}

如果使用BeginInvoke, MessageBox会在文本更新的同时弹出。如果使用调用, MessageBox将在3秒休眠后弹出。因此,显示异步(BeginInvoke)和同步(调用)调用的效果。

只是添加为什么以及何时使用Invoke()。

Invoke()和BeginInvoke()都将您指定的代码封送到调度程序线程。

但与BeginInvoke()不同的是,Invoke()会暂停线程,直到分派器执行你的代码。如果需要暂停异步操作,直到用户提供某种反馈,则可能需要使用Invoke()。

例如,您可以调用Invoke()来运行显示OK/Cancel对话框的代码片段。在用户单击按钮并且您的封送代码完成之后,invoke()方法将返回,您可以根据用户的响应进行操作。

请参阅c#第31章中的Pro WPF