Winform 通过Invoke改变UI

本文介绍在WinForms应用程序中如何安全地从子线程更新主线程中的UI元素,避免线程间操作无效异常。通过使用Invoke方法,确保控件状态更改不会与主线程发生冲突。

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

在Winfrom 程序中如果直接在子线程中更新主线程的UI会触发如下线程间操作无效异常。
异常如下图所示:
这里写图片描述

这是因为控件是在主线程中创建的,在子线程中改变控件状态可能会与主线程发生冲突。如果主线程正在重绘控件外观,而此时子线程要改变控件状态则会造成画面混乱。

解决方案
使用Invoke, Invoke方法会顺着控件树向上搜索直到找到创建控件的那个线程(通常是主线程),然后进入那个线程改变控件的外观,确保不发生线程冲突。写法如下:

private void proc()
{
   this.label1.Invoke(newEventHandler(delegate
       {
           this.label1.Text= "改變後UI";
       }));
}

Invoke方法需要创建一个委托。
对于不同控件都有相应的Invoke方法,可以使用主窗口的Invoke方法,将要改变的控件写在一起

private void proc()
{
      this.Invoke(new EventHandler(delegate
      {
          this.label1.Text= "改變後UI";
      }));
}

在C# 3.0及以后的版本中可以使用Lamda表达式,对于上面的匿名委托,可以使用Action封装方法,代码如下:

privatevoid proc()
{
   this.Invoke(newAction(()=>
   {
        this.label1.Text= "改變後UI";
   }));
}

完整代码如下:

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading;
using System.Windows.Forms;
using System.Security.Cryptography;

namespace WinformExercise
{
    public partial class Form1 : Form
    {
        public Form1()
        {
            InitializeComponent();
        }

        private void button1_Click(object sender, EventArgs e)
        {
            Thread t = new Thread(proc);
            t.Start();
        }

        private void proc()
        {
            this.Invoke(new EventHandler(delegate
            {
                this.label1.Text = "改變後UI";
            }));

        }
    }
}
### 如何在 WinForms 中使用 Invoke 方法 `Invoke` 方法用于确保线程安全的操作,特别是在多线程环境中更新 UI 组件时。当后台线程需要修改由其他线程拥有的控件属性或调用其方法时,应使用 `Invoke` 来调度该操作到拥有此控件的线程上执行。 #### 使用 Invoke 更新 UI 控件 下面是一个简单的例子来展示如何在线程中使用 `Invoke` 方法: ```csharp private void UpdateLabel(string newText) { if (label1.InvokeRequired) { label1.Invoke(new Action<string>(UpdateLabel), new object[] { newText }); } else { label1.Text = newText; } } ``` 这段代码展示了如何通过判断 `InvokeRequired` 属性决定是否需要跨线程访问,并利用 `Invoke` 将更改应用到 UI 线程[^1]。 #### 完整示例:异步任务中的 Invoke 应用 为了更全面理解 `Invoke` 的应用场景,这里提供了一个完整的案例,在其中启动一个新的线程模拟耗时的任务并定期报告进度给主线程上的 Label 控件: ```csharp using System.Threading; public partial class MainForm : Form { public MainForm() { InitializeComponent(); Thread thread = new Thread(DoWork); thread.Start(); } private void DoWork() { for (int i = 0; i <= 100; i++) { // Simulate work being done... Thread.Sleep(100); string message = $"Progress: {i}%"; // Safely update the UI control from another thread. UpdateLabel(message); } } private void UpdateLabel(string text) { if (this.label1.InvokeRequired) { this.label1.Invoke((MethodInvoker)(() => this.label1.Text = text)); } else { this.label1.Text = text; } } } ``` 上述代码片段说明了如何在一个新线程内完成工作的同时保持界面响应性,并且每当有新的进展时就通知主UI线程去刷新状态显示。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值