异步委托

本文介绍了在多线程环境下安全访问Windows窗体控件的方法,对比了线程安全与非线程安全的方式,并提供了具体的代码示例。

线程间操作无效: 从不是创建控件“listViewLog”的线程访问它。
   变量一般可以访问(不安全)
   控件不行。


访问 Windows 窗体控件本质上不是线程安全的。如果有两个或多个线程操作某一控件的状态,则可能会迫使该控件进入一种不一致的状态。还可能出

现其他与线程相关的 bug,包括争用情况和死锁。确保以线程安全方式访问控件非常重要。
.NET Framework 有助于在以非线程安全方式访问控件时检测到这一问题。在调试器中运行应用程序时,如果创建某控件的线程之外的其他线程试图调

用该控件,则调试器会引发一个 InvalidOperationException,并提示消息:“从不是创建控件 control name 的线程访问它。”

此异常在调试期间和运行时的某些情况下可靠地发生。强烈建议您在显示此错误信息时修复此问题。在调试以 .NET Framework 2.0 版之前的 .NET

Framework 编写的应用程序时,可能会出现此异常。

 

 

下面的代码示例演示如何从辅助线程以线程安全方式和非线程安全方式调用 Windows 窗体控件。它演示一种以非线程安全方式设置 控件的 属性的方法,还演示两种以线程安全方式设置 Text 属性的方法。

 

ContractedBlock.gifExpandedBlockStart.gifCode
using System;
using System.ComponentModel;
using System.Threading;
using System.Windows.Forms;

namespace CrossThreadDemo
{
    
public class Form1 : Form
    {
        
// This delegate enables asynchronous calls for setting
        
// the text property on a TextBox control.
        delegate void SetTextCallback(string text);

        
// This thread is used to demonstrate both thread-safe and
        
// unsafe ways to call a Windows Forms control.
        private Thread demoThread = null;

        
// This BackgroundWorker is used to demonstrate the 
        
// preferred way of performing asynchronous operations.
        private BackgroundWorker backgroundWorker1;

        
private TextBox textBox1;
        
private Button setTextUnsafeBtn;
        
private Button setTextSafeBtn;
        
private Button setTextBackgroundWorkerBtn;

        
private System.ComponentModel.IContainer components = null;

        
public Form1()
        {
            InitializeComponent();
        }

        
protected override void Dispose(bool disposing)
        {
            
if (disposing && (components != null))
            {
                components.Dispose();
            }
            
base.Dispose(disposing);
        }

        
// This event handler creates a thread that calls a 
        
// Windows Forms control in an unsafe way.
        private void setTextUnsafeBtn_Click(
            
object sender, 
            EventArgs e)
        {
            
this.demoThread = 
                
new Thread(new ThreadStart(this.ThreadProcUnsafe));

            
this.demoThread.Start();
        }

        
// This method is executed on the worker thread and makes
        
// an unsafe call on the TextBox control.
        private void ThreadProcUnsafe()
        {
            
this.textBox1.Text = "This text was set unsafely.";
        }

        
// This event handler creates a thread that calls a 
        
// Windows Forms control in a thread-safe way.
        private void setTextSafeBtn_Click(
            
object sender, 
            EventArgs e)
        {
            
this.demoThread = 
                
new Thread(new ThreadStart(this.ThreadProcSafe));

            
this.demoThread.Start();
        }

        
// This method is executed on the worker thread and makes
        
// a thread-safe call on the TextBox control.
        private void ThreadProcSafe()
        {
            
this.SetText("This text was set safely.");
        }

        
// This method demonstrates a pattern for making thread-safe
        
// calls on a Windows Forms control. 
        
//
        
// If the calling thread is different from the thread that
        
// created the TextBox control, this method creates a
        
// SetTextCallback and calls itself asynchronously using the
        
// Invoke method.
        
//
        
// If the calling thread is the same as the thread that created
        
// the TextBox control, the Text property is set directly. 

        
private void SetText(string text)
        {
            
// InvokeRequired required compares the thread ID of the
            
// calling thread to the thread ID of the creating thread.
            
// If these threads are different, it returns true.
            if (this.textBox1.InvokeRequired)
            {    
                SetTextCallback d 
= new SetTextCallback(SetText);
                
this.Invoke(d, new object[] { text });
            }
            
else
            {
                
this.textBox1.Text = text;
            }
        }

        
// This event handler starts the form's 
        
// BackgroundWorker by calling RunWorkerAsync.
        
//
        
// The Text property of the TextBox control is set
        
// when the BackgroundWorker raises the RunWorkerCompleted
        
// event.
        private void setTextBackgroundWorkerBtn_Click(
            
object sender, 
            EventArgs e)
        {
            
this.backgroundWorker1.RunWorkerAsync();
        }
        
        
// This event handler sets the Text property of the TextBox
        
// control. It is called on the thread that created the 
        
// TextBox control, so the call is thread-safe.
        
//
        
// BackgroundWorker is the preferred way to perform asynchronous
        
// operations.

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

        
#region Windows Form Designer generated code

        
private void InitializeComponent()
        {
            
this.textBox1 = new System.Windows.Forms.TextBox();
            
this.setTextUnsafeBtn = new System.Windows.Forms.Button();
            
this.setTextSafeBtn = new System.Windows.Forms.Button();
            
this.setTextBackgroundWorkerBtn = new System.Windows.Forms.Button();
            
this.backgroundWorker1 = new System.ComponentModel.BackgroundWorker();
            
this.SuspendLayout();
            
// 
            
// textBox1
            
// 
            this.textBox1.Location = new System.Drawing.Point(1212);
            
this.textBox1.Name = "textBox1";
            
this.textBox1.Size = new System.Drawing.Size(24020);
            
this.textBox1.TabIndex = 0;
            
// 
            
// setTextUnsafeBtn
            
// 
            this.setTextUnsafeBtn.Location = new System.Drawing.Point(1555);
            
this.setTextUnsafeBtn.Name = "setTextUnsafeBtn";
            
this.setTextUnsafeBtn.TabIndex = 1;
            
this.setTextUnsafeBtn.Text = "Unsafe Call";
            
this.setTextUnsafeBtn.Click += new System.EventHandler(this.setTextUnsafeBtn_Click);
            
// 
            
// setTextSafeBtn
            
// 
            this.setTextSafeBtn.Location = new System.Drawing.Point(9655);
            
this.setTextSafeBtn.Name = "setTextSafeBtn";
            
this.setTextSafeBtn.TabIndex = 2;
            
this.setTextSafeBtn.Text = "Safe Call";
            
this.setTextSafeBtn.Click += new System.EventHandler(this.setTextSafeBtn_Click);
            
// 
            
// setTextBackgroundWorkerBtn
            
// 
            this.setTextBackgroundWorkerBtn.Location = new System.Drawing.Point(17755);
            
this.setTextBackgroundWorkerBtn.Name = "setTextBackgroundWorkerBtn";
            
this.setTextBackgroundWorkerBtn.TabIndex = 3;
            
this.setTextBackgroundWorkerBtn.Text = "Safe BW Call";
            
this.setTextBackgroundWorkerBtn.Click += new System.EventHandler(this.setTextBackgroundWorkerBtn_Click);
            
// 
            
// backgroundWorker1
            
// 
            this.backgroundWorker1.RunWorkerCompleted += new System.ComponentModel.RunWorkerCompletedEventHandler(this.backgroundWorker1_RunWorkerCompleted);
            
// 
            
// Form1
            
// 
            this.ClientSize = new System.Drawing.Size(26896);
            
this.Controls.Add(this.setTextBackgroundWorkerBtn);
            
this.Controls.Add(this.setTextSafeBtn);
            
this.Controls.Add(this.setTextUnsafeBtn);
            
this.Controls.Add(this.textBox1);
            
this.Name = "Form1";
            
this.Text = "Form1";
            
this.ResumeLayout(false);
            
this.PerformLayout();

        }

        
#endregion


        [STAThread]
        
static void Main()
        {
            Application.EnableVisualStyles();
            Application.Run(
new Form1());
        }

    }
}

转载于:https://www.cnblogs.com/goodyao/archive/2008/09/03/1283043.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值