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

c#在work线程中怎样更新UI控件

最近笔者调试修改项目,碰到了c#在work线程中怎样更新UI控件中的场景,简单总结了下,主要有两个方法:

方法1:通过System.Windows.Application.Current.Dispatcher.Invoke来更新UI控件

 System.Windows.Application.Current.Dispatcher.Invoke(() =>
 {
     ConnectCheckBox.IsChecked = false;
 });

方法2:通过SynchronizationContext实现,具体细节如下:

SynchronizationContext.Post 函数:主要用于在 Windows Forms 或者 WPF 等 UI 相关的应用程序中,以异步的方式将一个委托(在这里是 SetTextSafePost 方法)投递到指定的同步上下文(m_SyncContext)中执行,下面详细展开解释其作用和原理。

1. 同步上下文(SynchronizationContext)的概念

在 Windows Forms 或者 WPF 应用程序中,UI 控件(如 ListViewTextBox 等)是线程不安全的,这意味着只有创建这些控件的线程(通常是主线程,也就是 UI 线程)才能够访问和修改它们。如果在其他线程中直接访问和修改 UI 控件,会抛出 InvalidOperationException 异常。

SynchronizationContext 类提供了一种机制,允许在不同线程之间安全地切换执行上下文,确保在访问 UI 控件时是在 UI 线程中进行的。在 Windows Forms 应用程序中,SynchronizationContext.Current 会返回当前线程的同步上下文,对于 UI 线程来说,这个同步上下文会确保委托在 UI 线程中执行。

2. m_SyncContext.Post 方法的作用

m_SyncContext.Post 方法用于将一个委托异步投递到指定的同步上下文中执行。它接受两个参数:

  • 第一个参数是一个 SendOrPostCallback 委托,这个委托指向要执行的方法(在这里是 SetTextSafePost 方法)。
  • 第二个参数是传递给委托方法的参数(在这里是 lvi,即 ListViewItem 对象)。

Post 方法会将委托放入同步上下文的消息队列中,然后立即返回,不会等待委托执行完成。同步上下文会在合适的时机(通常是 UI 线程处理消息队列时)从消息队列中取出委托并在 UI 线程中执行。

private void SetTextSafePost(object lvi)
{
    if (AlarmabsListView.Items.Count > 100)
    {
        AlarmabsListView.Items.RemoveAt(100);
    }
    ListViewItem temp = lvi as ListViewItem;
    AlarmabsListView.Items.Insert(0, temp);
}

private static SynchronizationContext m_SyncContext = null;

// 在某个地方初始化同步上下文,通常在 UI 线程中进行
m_SyncContext = SynchronizationContext.Current;

// ...

if (strJson.ToLower().Contains("\"testInfo\""))
{    
    ListViewItem lvi = new ListViewItem();
    lvi.Text = heartBeat.EventNotificationAlert.dataTime;
    lvi.SubItems.Add(heartBeat.EventNotificationAlert.eventType);
    lvi.SubItems.Add(strJson);

    m_SyncContext.Post(SetTextSafePost, lvi);
}

4. 与 Send 方法的区别

SynchronizationContext 类还有一个 Send 方法,它与 Post 方法的区别在于:

  • Post 方法是异步的,它会将委托放入消息队列中,然后立即返回,不会等待委托执行完成。
  • Send 方法是同步的,它会等待委托执行完成后才返回。

在处理 UI 控件时,通常建议使用 Post 方法,以避免阻塞当前线程。

相关文章:

  • Android 10.0 SystemUI状态栏去掉刘海屏功能实现
  • 26、web前端开发之CSS3(三)
  • day 15
  • GAUSSDB 分布式存储机制深度解析
  • 前端路由与导航:React Router 与 Vue Router
  • Redis 单线程
  • 计算机期刊推荐 | 工程技术, 计算机
  • 三极管原理及应用
  • Pytorch学习笔记(九)Learning PyTorch - Deep Learning with PyTorch: A 60 Minute Blitz
  • 华为HCIE学习指南,如何更好的学习HCIE?
  • AI-Sphere-Butler之如何使用腾讯云ASR语音识别服务
  • doris:查询熔断
  • 洛谷 P10516 数据结构 Solution
  • Maven工具学习使用(三)——坐标和依赖
  • 基于Python的自然语言处理系列(60):使用 LangChain 构建 Multi-Vector Retriever 进行文档检索
  • Redis部署详细步骤
  • 数字电子技术基础(三十三)——编码器
  • 数据库练习
  • 使用ZMQ和protobuf实现C++程序与Python程序的通信
  • 多行为推荐综述
  • 中俄就应对美加征所谓“对等关税”等问题进行深入交流
  • 乡村快递取件“跑腿费”屡禁不止?云南元江县公布举报电话
  • 60岁济南石化设计院党总支书记、应急管理专家李有臣病逝
  • 央行行长:债券市场“科技板”准备工作基本就绪,目前近百家市场机构计划发行超三千亿科技创新债
  • 象屿集团:对去化压力大、市场有效需求不足区域坚决暂停投资,打造多元上市路径
  • 前瞻|中俄元首今年将首次面对面会晤,专家:国际变局中构建更坚韧的合作架构