Control.Invoke 方法

定义

在拥有控件的基础窗口句柄的线程上执行委托。

重载

Invoke(Action)

在拥有控件的基础窗口句柄的线程上执行指定的委托。

Invoke(Delegate)

在拥有控件的基础窗口句柄的线程上执行指定的委托。

Invoke(Delegate, Object[])

在拥有控件的基础窗口句柄的线程上,使用指定的参数列表执行指定的委托。

Invoke<T>(Func<T>)

在拥有控件的基础窗口句柄的线程上执行指定的委托。

Invoke(Action)

在拥有控件的基础窗口句柄的线程上执行指定的委托。

public void Invoke (Action method);

参数

method
Action

一个委托,其中包含在控件的线程上下文中调用的方法。

适用于

Windows Desktop 9 和其他版本
产品 版本
Windows Desktop 6, 7, 8, 9

Invoke(Delegate)

在拥有控件的基础窗口句柄的线程上执行指定的委托。

public object Invoke (Delegate method);

参数

method
Delegate

一个委托,其中包含在控件的线程上下文中调用的方法。

返回

正在调用的委托的返回值,或者如果委托没有返回值,则 null

示例

下面的代码示例显示包含委托的控件。 委托封装了将项添加到列表框的方法,此方法在拥有窗体的基础句柄的线程上执行。 当用户单击按钮时,Invoke 运行委托。

/*
The following example demonstrates the 'Invoke(Delegate)' method of 'Control class.
A 'ListBox' and a 'Button' control are added to a form, containing a delegate
which encapsulates a method that adds items to the listbox. This function is executed
on the thread that owns the underlying handle of the form. When user clicks on button
the above delegate is executed using 'Invoke' method.


*/

using System;
using System.Drawing;
using System.Windows.Forms;
using System.Threading;

   public class MyFormControl : Form
   {
      public delegate void AddListItem();
      public AddListItem myDelegate;
      private Button myButton;
      private Thread myThread;
      private ListBox myListBox;
      public MyFormControl()
      {
         myButton = new Button();
         myListBox = new ListBox();
         myButton.Location = new Point(72, 160);
         myButton.Size = new Size(152, 32);
         myButton.TabIndex = 1;
         myButton.Text = "Add items in list box";
         myButton.Click += new EventHandler(Button_Click);
         myListBox.Location = new Point(48, 32);
         myListBox.Name = "myListBox";
         myListBox.Size = new Size(200, 95);
         myListBox.TabIndex = 2;
         ClientSize = new Size(292, 273);
         Controls.AddRange(new Control[] {myListBox,myButton});
         Text = " 'Control_Invoke' example";
         myDelegate = new AddListItem(AddListItemMethod);
      }
      static void Main()
      {
         MyFormControl myForm = new MyFormControl();
         myForm.ShowDialog();
      }
      public void AddListItemMethod()
      {
         String myItem;
         for(int i=1;i<6;i++)
         {
            myItem = "MyListItem" + i.ToString();
            myListBox.Items.Add(myItem);
            myListBox.Update();
            Thread.Sleep(300);
         }
      }
      private void Button_Click(object sender, EventArgs e)
      {
         myThread = new Thread(new ThreadStart(ThreadFunction));
         myThread.Start();
      }
      private void ThreadFunction()
      {
         MyThreadClass myThreadClassObject  = new MyThreadClass(this);
         myThreadClassObject.Run();
      }
   }

// The following code assumes a 'ListBox' and a 'Button' control are added to a form, 
// containing a delegate which encapsulates a method that adds items to the listbox.

   public class MyThreadClass
   {
      MyFormControl myFormControl1;
      public MyThreadClass(MyFormControl myForm)
      {
         myFormControl1 = myForm;
      }

      public void Run()
      {
         // Execute the specified delegate on the thread that owns
         // 'myFormControl1' control's underlying window handle.
         myFormControl1.Invoke(myFormControl1.myDelegate);
      }
   }

注解

委托类似于 C 或 C++ 语言中的函数指针。 委托封装对委托对象内方法的引用。 然后,可以将委托对象传递给调用所引用方法的代码,在编译时调用的方法可能未知。 与 C 或 C++ 中的函数指针不同,委托面向对象、类型安全且更安全。

如果当前控件的基础窗口句柄尚不存在,Invoke 方法将搜索控件的父链,直到找到具有窗口句柄的控件或窗体。 如果找不到适当的句柄,Invoke 方法将引发异常。 在调用期间引发的异常将传播回调用方。

备注

除了 InvokeRequired 属性之外,还有四个控件上的方法是线程安全的:InvokeBeginInvokeEndInvoke,如果已创建控件的句柄,CreateGraphics。 在后台线程上创建控件的句柄之前调用 CreateGraphics 可能会导致非法跨线程调用。 对于所有其他方法调用,应使用其中一个调用方法封送对控件线程的调用。

委托可以是 EventHandler的实例,在这种情况下,发送方参数将包含此控件,事件参数将包含 EventArgs.Empty。 委托也可以是 MethodInvoker的实例,也可以是采用 void 参数列表的任何其他委托。 对 EventHandlerMethodInvoker 委托的调用比对另一种类型的委托的调用要快。

备注

如果处理消息的线程不再处于活动状态,则可能会引发异常。

另请参阅

适用于

.NET Framework 4.8.1 和其他版本
产品 版本
.NET Framework 1.1, 2.0, 3.0, 3.5, 4.0, 4.5, 4.5.1, 4.5.2, 4.6, 4.6.1, 4.6.2, 4.7, 4.7.1, 4.7.2, 4.8, 4.8.1
Windows Desktop 3.0, 3.1, 5, 6, 7, 8, 9

Invoke(Delegate, Object[])

在拥有控件的基础窗口句柄的线程上,使用指定的参数列表执行指定的委托。

public object Invoke (Delegate method, object[] args);
public object Invoke (Delegate method, params object[] args);
public object Invoke (Delegate method, params object?[]? args);

参数

method
Delegate

一个委托给采用 args 参数中包含的相同数字和类型的参数的方法。

args
Object[]

要作为参数传递给指定方法的对象数组。 如果方法不采用任何参数,则可以 null 此参数。

返回

一个 Object,其中包含所调用的委托的返回值,或者如果委托没有返回值,则 null

实现

示例

下面的代码示例显示包含委托的控件。 委托封装了将项添加到列表框的方法,此方法使用指定的参数在拥有窗体的基础句柄的线程上执行。 当用户单击按钮时,Invoke 运行委托。

using System;
using System.Drawing;
using System.Windows.Forms;
using System.Threading;

   public class MyFormControl : Form
   {
      public delegate void AddListItem(String myString);
      public AddListItem myDelegate;
      private Button myButton;
      private Thread myThread;
      private ListBox myListBox;
      public MyFormControl()
      {
         myButton = new Button();
         myListBox = new ListBox();
         myButton.Location = new Point(72, 160);
         myButton.Size = new Size(152, 32);
         myButton.TabIndex = 1;
         myButton.Text = "Add items in list box";
         myButton.Click += new EventHandler(Button_Click);
         myListBox.Location = new Point(48, 32);
         myListBox.Name = "myListBox";
         myListBox.Size = new Size(200, 95);
         myListBox.TabIndex = 2;
         ClientSize = new Size(292, 273);
         Controls.AddRange(new Control[] {myListBox,myButton});
         Text = " 'Control_Invoke' example ";
         myDelegate = new AddListItem(AddListItemMethod);
      }
      static void Main()
      {
         MyFormControl myForm = new MyFormControl();
         myForm.ShowDialog();
      }
      public void AddListItemMethod(String myString)
      {
            myListBox.Items.Add(myString);
      }
      private void Button_Click(object sender, EventArgs e)
      {
         myThread = new Thread(new ThreadStart(ThreadFunction));
         myThread.Start();
      }
      private void ThreadFunction()
      {
         MyThreadClass myThreadClassObject  = new MyThreadClass(this);
         myThreadClassObject.Run();
      }
   }
   public class MyThreadClass
   {
      MyFormControl myFormControl1;
      public MyThreadClass(MyFormControl myForm)
      {
         myFormControl1 = myForm;
      }
      String myString;

      public void Run()
      {

         for (int i = 1; i <= 5; i++)
         {
            myString = "Step number " + i.ToString() + " executed";
            Thread.Sleep(400);
            // Execute the specified delegate on the thread that owns
            // 'myFormControl1' control's underlying window handle with
            // the specified list of arguments.
            myFormControl1.Invoke(myFormControl1.myDelegate,
                                   new Object[] {myString});
         }
      }
   }

注解

委托类似于 C 或 C++ 语言中的函数指针。 委托封装对委托对象内方法的引用。 然后,可以将委托对象传递给调用所引用方法的代码,在编译时调用的方法可能未知。 与 C 或 C++ 中的函数指针不同,委托面向对象、类型安全且更安全。

如果控件的句柄尚不存在,此方法将搜索控件的父链,直到找到具有窗口句柄的控件或窗体。 如果找不到适当的句柄,此方法将引发异常。 在调用期间引发的异常将传播回调用方。

备注

除了 InvokeRequired 属性之外,还有四个控件上的方法是线程安全的:InvokeBeginInvokeEndInvoke,如果已创建控件的句柄,CreateGraphics。 在后台线程上创建控件的句柄之前调用 CreateGraphics 可能会导致非法跨线程调用。 对于所有其他方法调用,应使用其中一个调用方法封送对控件线程的调用。

委托可以是 EventHandler的实例,在这种情况下,参数将取决于 args 值:

  • 如果未传递任何参数,发送方参数将包含此控件,事件参数将包含 EventArgs.Empty
  • 传递单个参数时,发送方参数将包含第一个参数元素,事件参数将包含 EventArgs.Empty
  • 如果传递了多个参数,发送方参数将包含来自 args的第一个元素,EventArgs 参数将包含第二个元素。

EventHandlerMethodInvoker 委托的调用比对另一种类型的委托的调用要快。

备注

如果处理消息的线程不再处于活动状态,则可能会引发异常。

另请参阅

适用于

.NET Framework 4.8.1 和其他版本
产品 版本
.NET Framework 1.1, 2.0, 3.0, 3.5, 4.0, 4.5, 4.5.1, 4.5.2, 4.6, 4.6.1, 4.6.2, 4.7, 4.7.1, 4.7.2, 4.8, 4.8.1
Windows Desktop 3.0, 3.1, 5, 6, 7, 8, 9

Invoke<T>(Func<T>)

在拥有控件的基础窗口句柄的线程上执行指定的委托。

public T Invoke<T> (Func<T> method);

类型参数

T

method的返回类型。

参数

method
Func<T>

在控件的线程上下文中调用的函数。

返回

T

正在调用的函数的返回值。

适用于

Windows Desktop 9 和其他版本
产品 版本
Windows Desktop 6, 7, 8, 9