聊透多线程编程-线程基础-3.C# Thread 如何从非UI线程直接更新UI元素

目录

1. 使用 Control.Invoke 或 Control.BeginInvoke(Windows Forms)

2. 使用 Dispatcher.Invoke 或 Dispatcher.BeginInvoke(WPF)

3. 使用 SynchronizationContext


 

桌面应用程序(如 Windows Forms 或 WPF)中,UI 操作必须由主线程(也称 UI 线程)执行。如果尝试从非 UI 线程直接更新 UI 元素,通常会引发异常或导致不可预测的行为。

Thread 类本身无法直接更新 UI,但可以通过以下方法将操作委托给 UI 线程来实现安全的 UI 更新。


1. 使用 Control.Invoke 或 Control.BeginInvoke(Windows Forms)

在 Windows Forms 应用程序中,可以使用 Control.Invoke 或 Control.BeginInvoke 方法将代码调度到 UI 线程。

示例代码:

using System;
using System.Reflection.Emit;
using System.Threading;
using System.Windows.Forms;
using static System.Net.Mime.MediaTypeNames;class Program : Form
{private Button button;private Label label;public Program(){button = new Button { Text = "Start Thread", Dock = DockStyle.Top };label = new Label { Text = "Waiting...", Dock = DockStyle.Fill };button.Click += Button_Click;Controls.Add(label);Controls.Add(button);}private void Button_Click(object sender, EventArgs e){Thread thread = new Thread(UpdateLabel);thread.Start();}private void UpdateLabel(){for (int i = 0; i < 10; i++){// 检查是否需要调用 Invokeif (label.InvokeRequired){// 使用 Invoke 将操作调度到 UI 线程label.Invoke(new Action(() => label.Text = $"Count: {i}"));}else{// 如果当前线程是 UI 线程,则直接更新label.Text = $"Count: {i}";}Thread.Sleep(500); // 模拟工作}}[STAThread]static void Main(){Application.EnableVisualStyles();Application.Run(new Program());}
}

解释:

  • InvokeRequired:检查当前线程是否是创建控件的线程(即 UI 线程)。如果不是,则需要通过 Invoke 或 BeginInvoke 调度到 UI 线程。
  • Invoke:同步执行指定的操作,等待操作完成后再继续。
  • BeginInvoke:异步执行指定的操作,不阻塞当前线程。

 

输出效果:
点击按钮后,label 的文本会每 500 毫秒更新一次,显示当前计数值。


2. 使用 Dispatcher.Invoke 或 Dispatcher.BeginInvoke(WPF)

在 WPF 应用程序中,可以使用 Dispatcher 对象将操作调度到 UI 线程。

示例代码:

using System;
using System.Reflection.Metadata;
using System.Threading;
using System.Windows;
using System.Windows.Controls;
using static System.Net.Mime.MediaTypeNames;class MainWindow : Window
{private Button button;private TextBlock textBlock;public MainWindow(){button = new Button { Content = "Start Thread" };textBlock = new TextBlock { Text = "Waiting..." };button.Click += Button_Click;var stackPanel = new StackPanel();stackPanel.Children.Add(button);stackPanel.Children.Add(textBlock);Content = stackPanel;}private void Button_Click(object sender, RoutedEventArgs e){Thread thread = new Thread(UpdateTextBlock);thread.Start();}private void UpdateTextBlock(){for (int i = 0; i < 10; i++){// 使用 Dispatcher 将操作调度到 UI 线程textBlock.Dispatcher.Invoke(() => textBlock.Text = $"Count: {i}");Thread.Sleep(500); // 模拟工作}}
}class Program
{[STAThread]static void Main(){var app = new Application();app.Run(new MainWindow());}
}

解释:

  • Dispatcher.Invoke:同步执行指定的操作,确保操作在 UI 线程上运行。
  • Dispatcher.BeginInvoke:异步执行指定的操作,不阻塞当前线程。

输出效果:

点击按钮后,TextBlock 的文本会每 500 毫秒更新一次,显示当前计数值。


3. 使用 SynchronizationContext

SynchronizationContext 是一种更通用的方式,适用于 Windows Forms 和 WPF,甚至其他框架(如 ASP.NET)。它允许你捕获当前线程的上下文,并在需要时将其用于调度操作。

示例代码:

using System;
using System.Reflection.Emit;
using System.Threading;
using System.Windows.Forms;
using static System.Net.Mime.MediaTypeNames;class Program : Form
{private Button button;private Label label;private SynchronizationContext _uiContext;public Program(){button = new Button { Text = "Start Thread", Dock = DockStyle.Top };label = new Label { Text = "Waiting...", Dock = DockStyle.Fill };button.Click += Button_Click;Controls.Add(label);Controls.Add(button);// 捕获 UI 线程的上下文_uiContext = SynchronizationContext.Current;}private void Button_Click(object sender, EventArgs e){Thread thread = new Thread(UpdateLabel);thread.Start();}private void UpdateLabel(){for (int i = 0; i < 10; i++){// 使用 SynchronizationContext 将操作调度到 UI 线程_uiContext.Post(_ => label.Text = $"Count: {i}", null);Thread.Sleep(500); // 模拟工作}}[STAThread]static void Main(){Application.EnableVisualStyles();Application.Run(new Program());}
}

解释:

  • SynchronizationContext.Current:捕获当前线程的上下文(通常是 UI 线程的上下文)。
  • Post:异步执行指定的操作。
  • Send:同步执行指定的操作。

输出效果:

点击按钮后,label 的文本会每 500 毫秒更新一次,显示当前计数值。

 

 

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/76100.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

TCP 和 UDP 可以使用同一个端口吗?

TCP 和 UDP 可以使用同一个端口吗&#xff1f; 前言 在深入探讨 TCP 和 UDP 是否可以使用同一个端口之前&#xff0c;我们首先需要理解网络通信的基本原理。网络通信是一个复杂的过程&#xff0c;涉及到多个层次的协议和机制。在 OSI 模型中&#xff0c;传输层是负责端到端数…

RVOS-2.基于NS16550a ,为os添加终端交互功能。

2.1 实验目的 为os添加uart功能&#xff0c;通过串口实现开发板与PC交互。 2.1 硬件信息 QEMU虚拟SoC含有 虚拟NS16550A设备 。 不同的地址线组合&#xff08;A2、A1、A0&#xff09;对应的读写模式和寄存器如下所示&#xff1a; 2.2 NS16550a 的初始化 线路控制寄存器&#…

java导入excel更新设备经纬度度数或者度分秒

文章目录 一、背景介绍二、页面效果三、代码0.pom.xml1.ImportDevice.vue2.ImportDeviceError.vue3.system.js4.DeviceManageControl5.DeviceManageUserControl6.Repeater7.FileUtils8.ResponseModel9.EnumLongitudeLatitude10.词条 四、注意点本人其他相关文章链接 一、背景介…

【力扣hot100题】(080)爬楼梯

让我们掌声恭迎动态规划的始祖—— 最基础的动态规划&#xff0c;原始方法是维护一个数组&#xff0c;每次记录到该阶梯的方案数量&#xff0c;每次的数量是到上一个阶梯的方案数量加上到上上一阶梯的方案数量&#xff0c;因为只有两种走法。 进阶可以优化空间复杂度&#xf…

CVE-2025-24813 漏洞全解析|Apache Tomcat 关键路径绕过与RCE

CVE-2025-24813 漏洞全解析&#xff5c;Apache Tomcat 关键路径绕过与RCE 作者:Factor .Poc作者:iSee857 CVE-2025-24813 漏洞全解析&#xff5c;Apache Tomcat 关键路径绕过与RCE一、漏洞概述二、影响版本三、漏洞原理&#x1f3af; 利用流程&#xff08;两步&#xff09;&am…

初识Linux:常见指令与权限的理解,以及相关衍生知识

目录 前言 关于linux的简介 代码开源 网络功能强大 系统工具链完整 一、Linux下的基本指令 1.ls指令 2.pwd指令 3.cd指令 4.whoami指令 5.touch指令 6.mkdir指令 7.rm指令 8.man指令 9.cp指令 10.mv指令 11.nano指令 12.cat指令 13.tac指令 14.more指令 15.less指令 16.head指令…

JVM虚拟机篇(七):JVM垃圾回收器全面解析与G1深度探秘及四种引用详解

JVM垃圾回收器全面解析与G1深度探秘及四种引用详解 JVM虚拟机&#xff08;七&#xff09;&#xff1a;JVM垃圾回收器全面解析与G1深度探秘及四种引用详解一、JVM有哪些垃圾回收器1. Serial回收器2. ParNew回收器3. Parallel Scavenge回收器4. Serial Old回收器5. Parallel Old回…

革新电销流程,数企云外呼开启便捷 “直通车”

在当今竞争激烈的商业环境中&#xff0c;电销作为一种重要的营销手段&#xff0c;依旧在企业的客户拓展与业务增长中扮演着关键角色。然而&#xff0c;传统电销流程常常面临诸多困扰&#xff0c;像是封卡封号风险、接通率不理想、客户开发与管理艰难以及销售考核复杂等问题&…

适合工程建筑行业的OA系统有什么推荐?

工程行业具有项目周期长、协作链条复杂等特性&#xff0c;传统管理模式下的 “人治”“纸质化” 弊端日益凸显。OA 系统作为数字化管理的核心载体&#xff0c;通过流程标准化、数据可视化&#xff0c;精准解决工程行业项目管理核心痛点。 泛微 e-office 深度聚焦工程场景&#…

车载刷写架构 --- ECU收到相同的blockSequenceCounter数据包的思考

我是穿拖鞋的汉子,魔都中坚持长期主义的汽车电子工程师。 老规矩,分享一段喜欢的文字,避免自己成为高知识低文化的工程师: 周末洗了一个澡,换了一身衣服,出了门却不知道去哪儿,不知道去找谁,漫无目的走着,大概这就是成年人最深的孤独吧! 旧人不知我近况,新人不知我过…

C++ RAII 的用途及业务代码实现案例

C RAII 的用途及业务代码实现案例 RAII 的核心概念 RAII (Resource Acquisition Is Initialization&#xff0c;资源获取即初始化) 是 C 的核心编程范式&#xff0c;其核心思想是&#xff1a; 资源获取与对象构造绑定资源释放与对象析构绑定利用 C 对象生命周期自动管理资源…

黑马 SpringAI+DeepSeek 实战:从对话机器人到企业级知识库的大模型开发全攻略

附完整代码 项目案例&#xff0c;3 天吃透大模型应用开发核心技术 需要完整项目学习视频以及源码的私信博主&#xff0c;谢谢~大家一起加油呐&#xff01;&#xff01; 01.认识AI和大模型 小结 AI的发展过程 符号主义 机器学习 深度学习——自然语言处理&#xff08;NLP…

共工新闻社与韩国新华报社达成合作

在当下媒体融合浪潮奔涌的时代背景下&#xff0c;大湾区经济网战略媒体香港共工新闻社与韩国新华报社顺利签署合作协议&#xff0c;携手为传播全球化进程以及海外华文媒体从单一媒体向多媒体的内涵拓展&#xff0c;乃至区域经济协同与文化融合发展贡献力量。 缔结友好华文媒体协…

嵌入式Linux驱动——3 总线设备驱动模型

目录 1.总线设备驱动模型 1.1 总线设备驱动模型 1.2 设备树 1.3 platform_device 和 platform_driver 的匹配规则 1.3.1 最先比较 1.3.2 然后比较 1.3.3 最后比较 2.LED 模板驱动程序的改造&#xff1a;总线设备驱动模型 1.总线设备驱动模型 在前面的 led 驱动程序中…

操作系统常用命令

逻辑卷创建及挂载步骤&#xff1a; vgcreate vg_app /dev/sda //在sda盘上创建vg_app卷组 lvcreate -L 50G -n lv_mysql vg_app //在vg_app卷组上创建逻辑卷lv_mysql mkfs.xfs /dev/vg_app/lv_mysql //对lv_mysql 逻辑卷创建文件系统 mkdir mysql //创建mysql目录 ech…

Git 的进阶功能和技巧

1、分支的概念和使用 1.1、什么是分支&#xff1f; 分支&#xff08;Branch&#xff09;是在版本控制中非常重要的概念。几乎所有版本控制系统都支持某种形式的分支。在 Git 中&#xff0c;分支是 Git 强大功能之一&#xff0c;它允许我们从主开发线分离出来&#xff0c;在不…

mapbox基础,加载F4Map二维地图

👨‍⚕️ 主页: gis分享者 👨‍⚕️ 感谢各位大佬 点赞👍 收藏⭐ 留言📝 加关注✅! 👨‍⚕️ 收录于专栏:mapbox 从入门到精通 文章目录 一、🍀前言1.1 ☘️mapboxgl.Map 地图对象1.2 ☘️mapboxgl.Map style属性二、🍀F4Map 简介2.1 ☘️技术特点2.2 ☘️核…

Conda使用方法详解

Conda是一个开源的包管理和环境管理系统&#xff0c;主要用于Python/R等科学计算领域&#xff0c;可以轻松管理不同项目的依赖关系。以下是Conda的详细使用方法&#xff1a; 一、安装与配置 1.安装Miniconda/Anaconda Miniconda是精简版&#xff0c;只包含conda和Python Ana…

Unity ViewportConstraint

一、组件功能概述 ViewportConstraint是一个基于世界坐标的UI边界约束组件&#xff0c;主要功能包括&#xff1a; 将UI元素限制在父容器范围内支持自定义内边距&#xff08;padding&#xff09;可独立控制水平和垂直方向的约束 二、实现原理 1. 边界计算&#xff08;世界坐…

代码随想录-动态规划24

leetcode-300-最长递增子序列 dp[i]表示i之前包括i的以nums[i]结尾的最长递增子序列的长度 dp[j]是(0,i-1)不包括i的以nums[i-1]结尾的最长递增子序列长度 int lengthOfLIS(int* nums, int numsSize) {if(numsSize < 1)return numsSize;int dp[numsSize];for(int i 0 ; i &…