Invoke的用法

Windows Forms 的控件是线程绑定 的,只能在创建它的UI 线程上访问 / 修改。

  • 你在后台线程(比如 Task.RunThread、VisionPro 回调线程)里直接操作控件,会抛出 Cross-thread operation not valid 异常。
  • Invoke 的作用就是:把操作控件的代码,"切回" UI 线程执行,避免跨线程异常。

它本质是一个消息转发器:后台线程把一个委托扔给 UI 线程,UI 线程空闲时再执行这个委托里的代码。


二、Invoke 最标准的用法(直接抄就能用)

1. 最通用的写法(支持所有控件)

复制代码
// 场景:在后台线程里更新TextBox1的文本
private void UpdateUI()
{
    // 后台线程里的代码
    Task.Run(() =>
    {
        string result = "检测完成";
        int count = 10;

        // 关键:用Invoke包裹所有控件操作
        textBox1.Invoke(new Action(() =>
        {
            textBox1.Text = result;
            textBox2.Text = count.ToString();
            richTextBox1.AppendText($"{DateTime.Now:HH:mm:ss} 检测结束\n");
        }));
    });
}

2. 封装成通用方法(避免到处写重复代码)

如果你项目里很多地方都要更新 UI,推荐封装一个通用方法:

复制代码
public static void InvokeIfRequired(Control control, Action action)
{
    if (control.InvokeRequired)
    {
        control.Invoke(action);
    }
    else
    {
        action();
    }
}

// 使用示例:
InvokeIfRequired(textBox1, () =>
{
    textBox1.Text = "OK";
    richTextBox1.AppendText("检测OK\n");
});

这个方法的好处是:不管当前是不是在 UI 线程,都能安全执行控件更新代码,不用每次都判断 InvokeRequired


三、你项目里的场景:VisionPro 回调里用 Invoke

结合你之前的 SetResultAndSave 方法,给你一个适配 VisionPro 回调的标准写法

复制代码
void SetResultAndSave(int a)
{
    string res = blob.Outputs["Output"].Value.ToString().Trim();
    string srcFile = files[a];
    string fileName = Path.GetFileName(srcFile);

    if (res == "OK")
    {
        zhong++;
        liang++;
        File.Copy(srcFile, Path.Combine(okPath, fileName), true);
    }
    else if (res == "NG")
    {
        zhong++;
        File.Copy(srcFile, Path.Combine(nqPath, fileName), true);
    }

    // 不管OK还是NG,统一用Invoke更新UI
    Invoke(new Action(() =>
    {
        richTextBox1.AppendText($"{DateTime.Now:HH:mm:ss} 第{a}次检测{res}\r\n");
        textBox2.Text = liang.ToString(); // OK数
        textBox1.Text = zhong.ToString(); // 总数
    }));
}

⚠️ 注意:不要把 File.Copy、计数这些耗时操作写进 Invoke ,否则会阻塞 UI 线程,界面卡死。Invoke 里只放控件操作代码!


四、最常见的错误用法(你之前踩过的坑)

❌ 错误 1:把耗时操作写进 Invoke

复制代码
// 错误!Thread.Sleep、文件读取都在UI线程执行,直接卡死界面
Invoke(new Action(() =>
{
    Thread.Sleep(1000); // 卡死UI线程
    File.ReadAllText("test.txt"); // 阻塞UI
    textBox1.Text = "OK";
}));

✅ 正确:耗时操作放后台线程,只把控件更新写进 Invoke

复制代码
Task.Run(() =>
{
    Thread.Sleep(1000); // 后台线程延时,不卡UI
    string text = File.ReadAllText("test.txt");

    textBox1.Invoke(new Action(() =>
    {
        textBox1.Text = text; // 只更新控件
    }));
});

五、进阶用法:带返回值的 Invoke(Invoke<T>)

如果你需要从 UI 线程获取数据,可以用带返回值的 Invoke

复制代码
// 场景:后台线程里获取textBox1的文本
string GetTextBoxText()
{
    if (textBox1.InvokeRequired)
    {
        return (string)textBox1.Invoke(new Func<string>(() => textBox1.Text));
    }
    else
    {
        return textBox1.Text;
    }
}

// 使用:
string text = GetTextBoxText();

六、总结:Invoke 用法口诀

  • 后台线程做耗时,控件更新用 Invoke
  • Invoke 里只写控件操作,不写延时、文件读写、循环
  • 统一封装通用方法,避免重复代码
  • 循环变量要捕获,避免闭包陷阱
  • 先判断控件是否存在,再调用 Invoke
相关推荐
techdashen1 小时前
What is maintenance, anyway?
开发语言·后端·rust
有什么事1 小时前
AI革命:云手机从脚本到智能体的跨越
人工智能·智能手机·自动化
o561-6o623o7鹿1 小时前
路,新生鼠适配器
人工智能
万法若空1 小时前
C/C++基本类型表示范围
c语言·开发语言·c++
2601_959477911 小时前
Vatee:外汇行情信息呈现与技术架构如何影响体验,给出一套细节
大数据·人工智能·安全·ux
yijianace1 小时前
Python爬虫实战:BooksToScrape 多线程爬取与图片下载
开发语言·爬虫·python
KaMeidebaby1 小时前
卡梅德生物技术快报|重组蛋白的表达和纯化:工艺调试全记录:大肠杆菌体系重组蛋白的表达和纯化参数标定(肠激酶轻链案例)
前端·人工智能·算法·数据挖掘·数据分析
kishu_iOS&AI1 小时前
LLM —— LangChain
人工智能·langchain
tedcloud1231 小时前
Supermemory部署教程:打造Agent记忆与RAG环境
服务器·人工智能·学习·自动化·powerpoint