C# 中的Invoke

本文介绍在.NET Framework的WinForm应用中如何安全地跨线程更新UI控件状态,避免出现“不能从不是创建该控件的线程调用它”的异常。讨论了使用Invoke方法的不同方式,包括针对不同类型的控件以及使用Lambda表达式的简洁写法。

本文转载连接: http://blog.youkuaiyun.com/u013985662/article/details/41146051?locationNum=3&fps=1


在用.NET Framework框架的WinForm构建GUI程序界面时,如果要在控件的事件响应函数中改变控件的状态,例如:某个按钮上的文本原先叫“打开”,单击之后按钮上的文本显示“关闭”,初学者往往会想当然地这么写:

void ButtonOnClick(object sender,EventArgs e)

{

    button.Text="关闭";

}

这样的写法运行程序之后,可能会触发异常,异常信息大致是“不能从不是创建该控件的线程调用它”。注意这里是“可能”,并不一定会触发该种异常。造成这种异常的原因在于,控件是在主线程中创建的(比如this.Controls.Add(...);),进入控件的事件响应函数时,是在控件所在的线程,并不是主线程。在控件的事件响应函数中改变控件的状态,可能与主线程发生线程冲突。如果主线程正在重绘控件外观,此时在别的线程改变控件外观,就会造成画面混乱。不过这样的情况并不总会发生,如果主线程此时在重绘别的控件,就可能逃过一劫,这样的写法可以正常通过,没有触发异常。

正确的写法是在控件响应函数中调用控件的Invoke方法(其实如果大家以前用过C++ Builder的话,也会找到类似Invoke那样的激活到主线程的函数)。Invoke方法会顺着控件树向上搜索,直到找到创建控件的那个线程(通常是主线程),然后进入那个线程改变控件的外观,确保不发生线程冲突。正确写法的示例如下:

void ButtonOnClick(object sender,EventArgs e)

{

    button.Invoke(new EventHandler(delegate

    {

        button.Text="关闭";

    }));

}

Invoke方法需要创建一个委托。你可以事先写好函数和与之对应的委托。不过,若想直观地在Invoke方法调用的时候就看到具体的函数,而不是到别处搜寻的话,上面的示例代码是不错的选择。

这样的写法有一个烦人的地方:对不同的控件写法不同。对于TextBox,要TextBoxObject.Invoke,对于Label,又要LabelObject.Invoke。有没有统一一点的写法呢?

主窗口类本身也有Invoke方法。如果你不想对不同的控件写法不一样,可以全部用this.Invoke:

void ButtonOnClick(object sender,EventArgs e)

{

    this.Invoke(new EventHandler(delegate

    {

        button.Text="关闭";

    }));

}

在C# 3.0及以后的版本中有了Lamda表达式,像上面这种匿名委托有了更简洁的写法。.NET Framework 3.5及以后版本更能用Action封装方法。例如以下写法可以看上去非常简洁:

void ButtonOnClick(object sender,EventArgs e)

{

    this.Invoke(new Action(()=>

    {

        button.Text="关闭";

    }));

}

以上写法往往充斥着WinForm构建的程序。

在微软新一代的界面开发技术WPF中,由于界面呈现和业务逻辑原生态地分开在两个线程中,所以控件的事件响应函数就不必Invoke了。但是,如果手动开辟一个新线程,那么在这个新线程中改变控件的外观,则还是要Invoke的。


C#中,Invoke是一个常用的方法,在委托和跨线程操作等场景中都有应用。 ### 使用方法 #### 委托中的Invoke使用 在C#委托里,Invoke方法可直接调用委托实例所指向的方法。示例代码如下: ```csharp using System; // 定义一个委托 delegate void MyDelegate(string message); class Program { static void PrintMessage(string message) { Console.WriteLine(message); } static void Main() { // 创建委托实例 MyDelegate del = PrintMessage; // 使用Invoke调用委托 del.Invoke("Hello, World!"); } } ``` 在上述代码中,先定义了一个委托`MyDelegate`,然后创建了该委托的实例`del`,并将`PrintMessage`方法赋值给它,最后使用`Invoke`方法调用委托实例,从而执行`PrintMessage`方法 [^2]。 #### 跨线程操作中的Invoke使用 在Windows Forms或WPF应用中,若要在非UI线程中更新UI元素,就需要使用`Invoke`方法。示例代码如下: ```csharp using System; using System.Threading; using System.Windows.Forms; namespace WindowsFormsApp1 { public partial class Form1 : Form { public Form1() { InitializeComponent(); } private void button1_Click(object sender, EventArgs e) { Thread thread = new Thread(DoWork); thread.Start(); } private void DoWork() { if (this.InvokeRequired) { this.Invoke(new Action(() => { // 更新UI元素 label1.Text = "Updated from another thread"; })); } else { label1.Text = "Updated from another thread"; } } } } ``` 该代码中,点击按钮会开启一个新线程执行`DoWork`方法。在`DoWork`方法里,先检查是否需要使用`Invoke`方法,如果需要,则使用`Invoke`方法来更新`label1`的文本内容 [^1]。 ### 应用场景 #### 跨线程更新UI 在Windows Forms或WPF应用中,UI元素只能由创建它们的线程(即UI线程)进行更新。当需要在其他线程中更新UI元素时,就必须使用`Invoke`方法来确保操作在UI线程上执行,避免出现线程安全问题 [^1]。 #### 委托调用 在委托中,`Invoke`方法可直接调用委托实例所指向的方法,这是一种简单直接的调用方式,适用于同步执行委托方法的场景 [^2]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值