1. 程式人生 > >C#多執行緒中訪問winform控制元件

C#多執行緒中訪問winform控制元件

方法一:System.Windows.Forms.Control.CheckForIllegalCrossThreadCalls = false; 

不推薦使用這種方式,禁止編譯器對跨執行緒訪問做檢查的方式實現。

方法二:使用delegate和invoke

private delegate void UpdateLabel2Delegate(string message);
        void UpdateLabel2(string message)
        {
            if (label1.InvokeRequired)
            {
                UpdateLabel2Delegate md = new UpdateLabel2Delegate(UpdateLabel2);
                label1.Invoke(md, new object[] { message });
		//label1.BeginInvoke(md, new object[] { message });
            }
            else
            {
                label1.Text = message;
            }
        }

或者:

void UpdateLabel2(string message)
{
 this.Invoke((EventHandler)delegate{ this.label1.Text = message;});
}

或者:(不使用委託方式)

private void button2_Click(object sender, EventArgs e)
        {
            Thread thread1 = new Thread(new ParameterizedThreadStart(UpdateLabel2));
            thread1.Start("更新Label");
        }

        private void UpdateLabel2(object str)
        {
            if (label2.InvokeRequired)
            {
                // 當一個控制元件的InvokeRequired屬性值為真時,說明有一個建立它以外的執行緒想訪問它
                Action<string> actionDelegate = (x) => { this.label2.Text = x.ToString(); };
                // 或者
                // Action<string> actionDelegate = delegate(string txt) { this.label2.Text = txt; };
                this.label2.Invoke(actionDelegate, str);
            }
            else
            {
                this.label2.Text = str.ToString();
            }
        }
方法三:使用BackgroundWorker元件
private void button4_Click(object sender, EventArgs e)
        {
            using (BackgroundWorker bw = new BackgroundWorker())
            {
                bw.RunWorkerCompleted += new RunWorkerCompletedEventHandler(bw_RunWorkerCompleted);
                bw.DoWork += new DoWorkEventHandler(bw_DoWork);
                bw.RunWorkerAsync("Tank");
            }         
        }
        void bw_DoWork(object sender, DoWorkEventArgs e)
        {       
            // 這裡是後臺執行緒, 是在另一個執行緒上完成的, 這裡是真正做事的工作執行緒
            // 可以在這裡做一些費時的,複雜的操作
            Thread.Sleep(5000);
            e.Result = e.Argument + "工作執行緒完成";
        }
        void bw_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e)
        {
            //這時後臺執行緒已經完成,並返回了主執行緒,所以可以直接使用UI控制元件了 
            this.label4.Text = e.Result.ToString(); 
        }