Invoke的用法(C#)

本文介绍了在多线程编程中如何安全地更新GUI界面显示,重点讲解了Invoke和BeginInvoke的区别及使用方法,通过一个具体示例展示了如何正确地在工作线程中更新界面元素。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

在多线程编程中,我们经常要在工作线程中去更新界面显示,而在多线程中直接调用界面控件的方法是错误的做法,Invoke BeginInvoke 就是为了解决这个问题而出现的,使你在多线程中安全的更新界面显示。

正确的做法是将工作线程中涉及更新界面的代码封装为一个方法,通过 Invoke 或者 BeginInvoke 去调用,两者的区别就是一个导致工作线程等待,而另外一个则不会。

而所谓的一面响应操作,一面添加节点永远只能是相对的,使 UI 线程的负担不至于太大而已,因为界面的正确更新始终要通过 UI 线程去做,我们要做的事情是在工作线程中包揽大部分的运算,而将对纯粹的界面更新放到 UI 线程中去做,这样也就达到了减轻 UI 线程负担的目的了。

举个简单例子说明下使用方法,比如你在启动一个线程,在线程的方法中想更新窗体中的一个TextBox..

using System.Threading;

//
启动一个线程
Thread thread=new Thread(new ThreadStart(DoWork));
thread.Start();

//
线程方法
private void DoWork()
{
     this.TextBox1.Text="
我是一个文本框";
}

如果你像上面操作,VS20052008里是会有异常的...

正确的做法是用Invoke/BeginInvoke

using System.Threading;
namespace test
{
    public partial class Form1 : Form
    {
        public delegate void MyInvoke(string str1,string str2);
        public Form1()
        {
            InitializeComponent();

 }
        public void DoWork()
        {
             MyInvoke mi = new MyInvoke(UpdateForm);
             this.BeginInvoke(mi, new Object[] {"
我是文本框","haha"});
        }
        public void UpdateForm(string param1,string parm2)
        {
            this.textBox1.Text = param1+parm2;
        }

        private void button1_Click(object sender, EventArgs e)
        {
             Thread thread = new Thread(new ThreadStart(DoWork));
             thread.Start();
        }
    }
}
注意代理的使用!

### C# Invoke 方法详解 #### 调用委托的方法 `Invoke` 是用于同步调用委托的一个重要方法。当需要在一个线程上调用另一个线程上的方法时,此功能特别有用。例如,在Windows Forms应用程序中更新UI控件通常需要通过 `Invoke` 来实现跨线程操作。 下面是一个简单的例子展示了如何利用 `Action` 委托以及 `Invoke` 方法来改变按钮的文字属性: ```csharp void ButtonOnClick(object sender, EventArgs e) { this.Invoke(new Action(() => { button.Text = "关闭"; })); } ``` 这段代码创建了一个新的 `Action` 对象并立即执行它,从而安全地修改了 UI 控件的内容[^1]。 #### 异步调用 BeginInvoke 和 EndInvoke 对于希望以非阻塞的方式运行某些逻辑的情况,则可以考虑使用 `BeginInvoke` 及其对应的完成通知机制——`EndInvoke`。这允许程序继续处理其他任务而不必等待被调用的操作结束。 假设有一个耗时较长的任务想要异步执行,可以通过如下方式设置: ```csharp // 定义一个接受整数参数返回字符串结果的委托 public delegate string LongRunningTaskDelegate(int input); private void StartLongTask() { var taskDelegate = new LongRunningTaskDelegate(LongRunningMethod); IAsyncResult result = taskDelegate.BeginInvoke(42, asyncResult => { try { // 当异步操作完成后获取最终的结果 string outcome = taskDelegate.EndInvoke(asyncResult); MessageBox.Show(outcome); } catch (Exception ex) { MessageBox.Show(ex.Message); } }, null); } ``` 这里展示的是启动长时间运行的方法,并在其结束后显示消息框告知用户[^2]。 #### P/Invoke 的应用实例 除了管理多线程间的通信外,`Invoke` 还可以在平台调用服务(P/Invoke)场景下发挥作用。比如要从托管环境访问未托管库中的函数,就需要先声明合适的委托类型作为接口桥梁。 以下是连接到外部C++动态链接库(DLL),并通过注册回调函数来进行数据交换的例子: ```csharp using System; using System.Runtime.InteropServices; class Program { // 定义回调函数的委托 public delegate void Callback(int value); [DllImport("MyCppLibrary.dll", CallingConvention = CallingConvention.Cdecl)] public static extern void RegisterCallback(Callback cb); static void Main(string[] args) { // 实例化回调函数 Callback callback = new Callback(PrintValue); // 注册回调给DLL内部使用 RegisterCallback(callback); } static void PrintValue(int value) { Console.WriteLine($"Callback value: {value}"); } } ``` 上述片段说明了怎样借助于 `delegate` 结合 `Invoke` 技术有效地实现了不同编程语言之间的协作工作流[^3]。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值