处理跨线程更新Winform UI控件

https://www.cnblogs.com/marshal-m/p/3201051.html

C#Winform编程中,跨线程直接更新UI控件的做法是不正确的,会时常出现“线程间操作无效: 从不是创建控件的线程访问它”的异常。处理跨线程更新Winform UI控件常用的方法有4种:
1. 通过UI线程的SynchronizationContext的Post/Send方法更新;
2. 通过UI控件的Invoke/BeginInvoke方法更新;

3. 通过BackgroundWorker取代Thread执行异步操作;
4. 通过设置窗体属性,取消线程安全检查来避免"跨线程操作异常"(非线程安全,建议不使用)。

 

Send() 是简单的在当前线程上去调用委托来实现(同步调用)。也就是在子线程上直接调用UI线程执行,等UI线程执行完成后子线程才继续执行。
 Post() 是在线程池上去调用委托来实现(异步调用)。这是子线程会从线程池中找一个线程去调UI线程,子线程不等待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.Tasks;
using System.Windows.Forms;
using System.Threading;

namespace WindowsFormsApp1
{
    public struct Item
    {
        public int id;
        public string name;
    };
    public partial class Form1 : Form
    {
        SynchronizationContext m_SyncContext = null;
        Thread demoThread;
        public Item item;
        public Form1()
        {
            InitializeComponent();
            m_SyncContext = SynchronizationContext.Current;
        }

        private void ThreadProcSafePost()
        {
            //...执行线程任务

            //在线程中更新UI(通过UI线程同步上下文m_SyncContext)

            item.id = 1;
            item.name = "abc";
            m_SyncContext.Post(SetTextSafePost, item);
            Thread.Sleep(3000);
            item.id = 2;
            item.name = "def";
            m_SyncContext.Post(SetTextSafePost, item);

            //...执行线程其他任务
        }
        private void SetTextSafePost(object item)
        {
            this.label1.Text = ((Item)item).name+ ((Item)item).id.ToString();
        }

        private void button1_Click(object sender, EventArgs e)
        {
            demoThread = new Thread(new ThreadStart(this.ThreadProcSafePost));
            this.demoThread.Start();
        }
    }
}
 

你可能感兴趣的:(c#)