Skip to content

C#跨线程操作控件的线程安全方法

在C#中,经常用到这样一个场景,Windows Form程序启动一个工作者线程执行一部分工作,这样做是为了避免速度慢的工作如果直接调用会使得主Form停止响应一段时间。 既然启动了线程,就避免不了线程之间数据传递的事情,相信你有很多种办法能解决,总之注意同步和互斥操作就好。我想说的是,工作线程处理中可能想操作某个主线程的Windows Form的Control,比如按钮,ListView等等更新工作状态之类,直接控制是不行的,不能够跨线程操作另一个线程创建的Windows Form控件。要使用委托去调用。

using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; using System.Drawing; using System.Text; using System.Windows.Forms; using System.Threading;

namespace JPGCompact { public partial class MainForm : Form { // 定义委托 private delegate void DelegateWriteResult(string file, bool result);

    // 与定义的委托签名相同的函数,操作主线程控件
    private void WriteResult(string fileName, bool result)
    {
        if (result)
        {
            ListViewItem thisListItem = new ListViewItem();
            thisListItem.ForeColor = Color.White;
            thisListItem.BackColor = Color.DarkGreen;
            thisListItem.SubItems.Text = fileName;
            thisListItem.SubItems.Add("成功");
            lvResultList.Items.Add(thisListItem);
        }
        else
        {
            ListViewItem thisListItem = new ListViewItem();
            thisListItem.ForeColor = Color.White;
            thisListItem.BackColor = Color.Red;
            thisListItem.SubItems.Text = fileName;
            thisListItem.SubItems.Add("失败");
            lvResultList.Items.Add(thisListItem);
        }
    }

    // 启动线程
    private void btnStart_Click(object sender, EventArgs e)
    {
        Thread workThread = new Thread(new ThreadStart(CompressAll));
        // 设置为背景线程,主线程一旦推出,该线程也不等待而立即结束
        workThread.IsBackground = true;
        workThread.Start();
    }

    // 线程执行函数
    private void CompressAll()
    {
        // 判断是否需要Invoke,多线程时需要
        if (this.InvokeRequired)
        {
            // 通过委托调用写主线程控件的程序,传递参数放在object数组中
            this.Invoke(new DelegateWriteResult(WriteResult),
                            new object[] { item, true });
        }
        else
        {
            // 如果不需要委托调用,则直接调用
            this.WriteResult(item, true);
        }
    }
}

}