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

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

 我們在做winform應用的時候,大部分情況下都會碰到使用多執行緒控制介面上控制元件資訊的問題。然而我們並不能用傳統方法來做這個問題,下面我將詳細的介紹。

首先來看傳統方法:

複製程式碼
 1 public partial class Form1 : Form
2 {
3 public Form1()
4 {
5 InitializeComponent();
6 }
7
8 private void Form1_Load(object sender, EventArgs e)
9 {
10 Thread thread = new
Thread(ThreadFuntion);
11 thread.IsBackground = true;
12 thread.Start();
13 }
14
15 private void ThreadFuntion()
16 {
17 while (true)
18 {
19 this.textBox1.Text = DateTime.Now.ToString();
20 Thread.Sleep(1000);
21 }
22 }
23 }
複製程式碼

  執行這段程式碼,我們會看到系統丟擲一個異常

:Cross-thread operation not valid:Control 'textBox1' accessed from a thread other than the thread it was created on . 這是因為.net 2.0以後加強了安全機制,不允許在winform中直接跨執行緒訪問控制元件的屬性。那麼怎麼解決這個問題呢,下面提供幾種方案。

第一種方案,我們在Form1_Load()方法中加一句程式碼:

複製程式碼
1 private void Form1_Load(object sender, EventArgs e)
2 {
3 Control.CheckForIllegalCrossThreadCalls = false;

4 Thread thread = new Thread(ThreadFuntion);
5 thread.IsBackground = true;
6 thread.Start();
7 }
複製程式碼

  加入這句程式碼以後發現程式可以正常運行了。這句程式碼就是說在這個類中我們不檢查跨執行緒的呼叫是否合法(如果沒有加這句話執行也沒有異常,那麼說明系統以及 預設的採用了不檢查的方式)。然而,這種方法不可取。我們檢視CheckForIllegalCrossThreadCalls 這個屬性的定義,就會發現它是一個static的,也就是說無論我們在專案的什麼地方修改了這個值,他就會在全域性起作用。而且像這種跨執行緒訪問是否存在異 常,我們通常都會去檢查。如果專案中其他人修改了這個屬性,那麼我們的方案就失敗了,我們要採取另外的方案。

下面來看第二種方案,就是使用delegate和invoke來從其他執行緒中控制控制元件資訊。執行下述程式碼,我們可以看到問題已經被解決了,通過等待非同步,我們就不會總是持有主執行緒的控制,這樣就可以在不發生跨執行緒呼叫異常的情況下完成多執行緒對winform多執行緒控制元件的控制了。

複製程式碼
 1 public partial class Form1 : Form
2 {
3 private delegate void FlushClient();//代理 4 public Form1()
5 {
6 InitializeComponent();
7 }
8
9 private void Form1_Load(object sender, EventArgs e)
10 {
11 Thread thread = new Thread(CrossThreadFlush);
12 thread.IsBackground = true;
13 thread.Start();
14 }
15
16 private void CrossThreadFlush()
17 {
18 while (true)
19 {
20 //將sleep和無限迴圈放在等待非同步的外面21 Thread.Sleep(1000);
22 ThreadFunction();
23 }
24 }
25
26 private void ThreadFunction()
27 {
28 if (this.textBox1.InvokeRequired)//等待非同步29 {
30 FlushClient fc = new FlushClient(ThreadFunction);
31 this.Invoke(fc);//通過代理呼叫重新整理方法32 }
33 else
34 {
35 this.textBox1.Text = DateTime.Now.ToString();
36 }
37 }
38 }
複製程式碼


轉自:http://www.7747.net/kf/201103/84569.html