当前位置: 首页 > article >正文

winform跨线程更新界面

前言:

大家好,我是上位机马工,硕士毕业4年年入40万,目前在一家自动化公司担任软件经理,从事C#上位机软件开发8年以上!我们在开发C#程序的时候,有时候需要在非Ui主线程更新界面,为了避免跨线程更新ui的异常,下面进行详细介绍如何实现这一需求!

1、报错代码

下面的代码中的this.Text指的是一个winform的窗体,开启Task执行下面的代码以后直接报错,提示线程间操作无效,这是因为在WinForms应用程序中,UI元素(如控件)通常只能在创建它们的线程(通常是主线程或UI线程)上进行操作。如果你尝试从另一个线程更新UI元素,将会引发跨线程操作异常(InvalidOperationException)。为了安全地从非UI线程更新UI,你需要使用Invoke或BeginInvoke方法将更新操作封送回UI线程。

private void Test()
        {
            for(int i=0;i<int.MaxValue;i++)
            {
                this.Text  = i.ToString();
                Thread.Sleep(1000);
            }
        }
   Task.Run(Test);

在这里插入图片描述

2、偷懒方法(不推荐)

在程序开始执行前设置下面的属性,意思就是不检查非UI线程访问UI,这样设置以后程序也不会报错,但是从众多查阅的资料可以得出该方法并不可靠,在多个线程同时并发访问控件时,可能会导致死锁,数据不一致等异常情况,所以并不推荐使用。

 CheckForIllegalCrossThreadCalls = false;


3、正确方法(推荐)

3.1 control.Invoke(一般都是采用这种方式)

下面的代码中this代表当前form窗体,Invoke方法里面传入的是一个委托,这里Action就是一个委托,Action委托绑定的方法通过lamada表达式实现,lamada表达式绑定的内容就是更新 this.Text的值。 control.Invoke的作用就是在创建控件的基础句柄所在线程上执行委托,也就是control.Invoke里面的委托更新ui的操作会被切换到ui线程执行,虽然调用 control.Invoke的代码不在ui线程,而且必须是ui线程的代码执行完了以后,才会重新切换到调用线程。

      private void Test()
        {
            this.Invoke(new Action(() =>
            {
                this.Text = "100";
                Console.WriteLine("委托执行完成");
                Thread.Sleep(1000);
            }));
           
            Console.WriteLine("Test方法执行完成");
        }
   Task.Run(Test);

输出:

委托执行完成
Test方法执行完成

从上面的输出可以看出 this.Invoke绑定的委托在ui线程执行完成以后,才返回到调用线程执行Console.WriteLine(“Test方法执行完成”);,这也称为同步调用。

3.2 control.BeginInvoke

下面的代码中this代表当前form窗体,BeginInvoke 方法里面传入的是一个委托,这里Action就是一个委托,Action委托绑定的方法通过lamada表达式实现,lamada表达式绑定的内容就是更新 this.Text的值。 control.BeginInvoke 的作用就是在创建控件的基础句柄所在线程上异步执行委托,也就是control.BeginInvoke里面的委托更新ui的操作会被切换到ui线程执行,同时BeginInvoke方法立即返回,接着执行调用线程后面的代码,而不是等到BeginInvoke绑定的委托执行完成以后才执行。

   private void Test()
        {
            this.BeginInvoke (new Action(() =>
            {
                this.Text = "100";
                Console.WriteLine("委托执行完成");
                Thread.Sleep(1000);
            }));
            Console.WriteLine("Test方法执行完成");
        }
   Task.Run(Test);

输出:

Test方法执行完成
委托执行完成

从上面的输出可以看出 this.BeginInvoke绑定的委托在ui线程执行的同时,也在调用线程执行Console.WriteLine(“Test方法执行完成”);,也就是说这两个线程并行执行,这也称为异步调用。

总结:对于winform跨线程更新界面推荐使用control.Invoke、 control.BeginInvoke,至于这两种方式分别在以下情况使用:
1)control.Invoke
对于程序执行速度要求不高的场合,推荐使用,这也是一般软件都使用的方式。
2)control.BeginInvoke
对于程序执行速度要求很高的场合,推荐使用,但是也有缺点,缺点就是具体界面在何时刷新我们是不知道的。

作者介绍

马工2017年硕士毕业,一直从事上位机软件开发工作,在我工作的第四年年薪突破了40万+,为了帮助跟我一样从底层出身的上位机软件工程师早日达到高级工程师的水平,早日找到30万+的工作,我根据多年项目经验,总结出了一系列可直接用于项目的C#上位机实战教程推荐给大家,目前在CSDN已经超过一千人订阅,如果你不甘贫庸,想像我一样早日拿到高薪,马工强烈推荐你早日学这套教程,雷军曾说这个世界上有99%的问题别人都遇到过,你要做的不是闷头干!而是找这个领域的专家问一下,这是最快速提升自己的方法!

年入30万+C#上位机实战必备教程推荐(点击下方链接即可访问文章)

1、《C#串口通信从入门到精通》
2、《C#与PLC通信从入门到精通 》
3、《C# Modbus通信从入门到精通》
4、《C#Socket通信从入门到精通 》
5、《C# MES通信从入门到精通》
6、《winform控件从入门到精通》


http://www.kler.cn/a/421596.html

相关文章:

  • Vue3 使用inject 获取provide 发布的响应式数据动态更新失败问题解决
  • 嵌入式硬件实战提升篇(三)商用量产电源设计方案 三路电源输入设计 电源管理 多输入供电自动管理 DCDC降压
  • Golang教程第24篇(语言接口)
  • 为什么编程语言会设计不可变的对象?字符串不可变?NSString *s = @“hello“变量s是不可变的吗?Rust内部可变性的意义?
  • uniapp在App端引用echarts组件,解决无法渲染formatter问题
  • pytest(二)excel数据驱动
  • 【Unity基础】Unity中Transform.forward的详解与应用
  • Spring Boot集成Spring Security:深入探索授权机制
  • 《山海经》:北山
  • 显卡(Graphics Processing Unit,GPU)架构详细解读
  • 故障诊断 | Transformer-LSTM组合模型的故障诊断(Matlab)
  • 高级java每日一道面试题-2024年12月02日-JVM篇-虚拟机为什么使用元空间替换了永久代?
  • 【C++boost::asio网络编程】有关异步Server样例以及伪闭包延长连接生命周期方法的笔记
  • react-router-dom 快速上手
  • 最小有向包围盒——2D平面
  • 【机器学习】CatBoost 模型实践:回归与分类的全流程解析
  • commitlint——Git提交规范
  • HTMLCSS 创意工坊:卡片网格的鼠标魔法秀
  • dns实验3:主从同步-完全区域传输
  • 蓝桥杯准备训练(lesson1,c++方向)
  • WebGL vendor [显卡]指纹
  • getchar()
  • L16.【LeetCode笔记】前序遍历
  • tp6 合成两个pdf文件(附加pdf或者替换pdf)
  • 力扣hot100道【贪心算法后续解题方法心得】(三)
  • idea的version control