如何:对 Windows 窗体控件进行线程安全调用

序:

访问 Windows 窗体控件本质上不是线程安全的。如果有两个或多个线程操作某一控件的状态,则可能会迫使该控件进入一种不一致的状态。还可能出现其他与线程相关的 bug,包括争用情况和死锁。确保以线程安全方式访问控件非常重要。

 

.NET Framework 有助于在以非线程安全方式访问控件时检测到这一问题。在调试器中运行应用程序时,如果创建某控件的线程之外的其他线程试图调用该控件,则调试器会引发一个 InvalidOperationException ,并显示以下消息:“从不是创建控件控件名称 的线程访问它。”

 

两种线程安全的访问方法:

 

对 Windows 窗体控件进行线程安全调用

  1. 查询控件的 InvokeRequired 属性。

  2. 如果 InvokeRequired 返回 true,则使用实际调用控件的委托来调用 Invoke 。

  3. 如果 InvokeRequired 返回 false,则直接调用控件。

示例:

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 WindowsFormsApplication2
{
    public partial class Form1 : Form
    {
        public Form1()
        {
            InitializeComponent();
        }
        delegate void Del();
        private void MyThread1()
        {
            if (textBox1.InvokeRequired)
            {
                Del d = new Del(delegate() { textBox1.Text += "线程一执行了一次/r/n"; });
                this.Invoke(d);
            }      
        }

        private void MyThread2()
        {
            if (textBox2.InvokeRequired)
            {
                Del d = new Del(delegate() { textBox2.Text += "线程二执行了一次/r/n"; });
                this.Invoke(d);
            }
        }

        private void button1_Click(object sender, EventArgs e)
        {
            Thread.Sleep(100);
            int i = 0;
            while(i<20)
            {
            Thread t = new Thread(new ThreadStart(MyThread1));
            t.Start();          
            i++;       
            }
        }

        private void button2_Click(object sender, EventArgs e)
        {
            Thread.Sleep(100);
            int i = 0;
            while (i < 20)
            {
                Thread t = new Thread(new ThreadStart(MyThread2));
                t.Start();
                i++;        
            }
        }

        private void Form1_Load(object sender, EventArgs e)
        {

        }
    }
}

使用 BackgroundWorker 进行的线程安全调用

在应用程序中实现多线程的首选方式是使用 BackgroundWorker 组件。 BackgroundWorker 组件使用事件驱动模型实现多线程。辅助线程运行 DoWork 事件处理程序,创建控件的线程运行 ProgressChanged 和 RunWorkerCompleted 事件处理程序。注意不要从 DoWork 事件处理程序调用您的任何控件。

下面的代码示例不异步执行任何工作,因此没有 DoWork 事件处理程序的实现。 TextBox 控件的 Text 属性在 RunWorkerCompleted 事件处理程序中直接设置。

示例:
private void setTextBackgroundWorkerBtn_Click(
object sender,
EventArgs e)
{
    this.backgroundWorker1.RunWorkerAsync();
}

  private void backgroundWorker1_RunWorkerCompleted(
object sender,
RunWorkerCompletedEventArgs e)
{
    this.textBox1.Text = "This text was set safely by BackgroundWorker.";
}


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值