在使用C#的過程中,難免會用到多線程,而用多線程之后,線程如何與界面交互則是一個非常頭疼的問題。其實不僅僅是界面,一般情況下,我們往往需要獲得線程的一些信息來確定線程的狀態。比較好的方式是用委托實現,看例子:
注:本例利用委托和跨線程訪問技術,用界面上的兩個label控件實時顯示線程的執行次數。網上雖然有很多這方面的文章,但是過於簡略,說明很少,剛剛接觸這方面的程序員很難理解,故寫此文。
TestClass類:
class TestClass { //聲明一個delegate(委托)類型:testDelegate,該類型可以搭載返回值為空,參數只有一個(long型)的方法。 public delegate void testDelegate(long i); //聲明一個testDelegate類型的對象。該對象代表了返回值為空,參數只有一個(long型)的方法。它可以搭載N個方法。 public testDelegate mainThread; /// <summary> /// 測試方法 /// </summary> public void testFunction() { long i = 0; while(true) { i++; mainThread(i); //調用委托對象 Thread.Sleep(1000); //線程等待1000毫秒 } } }
winform界面代碼:
/// <summary> /// 按鈕單擊事件 /// </summary> /// <param name="sender"></param> /// <param name="e"></param> private void button1_Click(object sender, EventArgs e) { //創建TestClass類的對象 TestClass testclass = new TestClass(); //在testclass對象的mainThread(委托)對象上搭載兩個方法,在線程中調用mainThread對象時相當於調用了這兩個方法。 testclass.mainThread = new TestClass.testDelegate(refreshLabMessage1); testclass.mainThread += new TestClass.testDelegate(refreshLabMessage2); //創建一個無參數的線程,這個線程執行TestClass類中的testFunction方法。 Thread testclassThread = new Thread(new ThreadStart(testclass.testFunction)); //啟動線程,啟動之后線程才開始執行 testclassThread.Start(); } /// <summary> /// 在界面上更新線程執行次數 /// </summary> /// <param name="i"></param> private void refreshLabMessage1(long i) { //判斷該方法是否被主線程調用,也就是創建labMessage1控件的線程,當控件的InvokeRequired屬性為ture時,說明是被主線程以外的線程調用。如果不加判斷,會造成異常 if (this.labMessage1.InvokeRequired) { //再次創建一個TestClass類的對象 TestClass testclass = new TestClass(); //為新對象的mainThread對象搭載方法 testclass.mainThread = new TestClass.testDelegate(refreshLabMessage1); //this指窗體,在這調用窗體的Invoke方法,也就是用窗體的創建線程來執行mainThread對象委托的方法,再加上需要的參數(i) this.Invoke(testclass.mainThread,new object[] {i}); } else { labMessage1.Text = i.ToString(); } } /// <summary> /// 在界面上更新線程執行次數 /// </summary> /// <param name="i"></param> private void refreshLabMessage2(long i) { //同上 if (this.labMessage2.InvokeRequired) { //再次創建一個TestClass類的對象 TestClass testclass = new TestClass(); //為新對象的mainThread對象搭載方法 testclass.mainThread = new TestClass.testDelegate(refreshLabMessage2); //this指窗體,在這調用窗體的Invoke方法,也就是用窗體的創建線程來執行mainThread對象委托的方法,再加上需要的參數(i) this.Invoke(testclass.mainThread, new object[] { i }); } else { labMessage2.Text = i.ToString(); } }
執行效果:

說明:
為了便於大家理解,我寫了很詳細的注釋。在這還要說明一下,因為這里邊有些“莫名其妙”的地方。
l 如何創建線程就不廢話了,一看就懂。
l public delegate void testDelegate(long i);這句話是創建了一個委托,名字是testDelegate,指定了委托的類型,什么返回值啦、什么參數啦,可以把testDelegate理解為一個類,一個規范;publictestDelegate mainThread;這句話當然就是創建testDelegate類的對象了,真正搭載方法的是mainThread對象,它可以搭載N個方法,順序執行。如何搭載捏?看這句話:testclass.mainThread= new TestClass.testDelegate(refreshLabMessage1);這句話是給testclass對象中的mainThread對象搭載方法,但是后邊的new比較難以理解。大家都知道,new這個關鍵字就是用來創建對象的,剛剛已經提醒大家把委托看成一個類,因此這new的是testDelegate這個委托,而不是TestClass(一定要看清了,如果是new的TestClass,要在TestClass后加括號的,后邊接的是方法,而testDelegate明顯不是方法,因此會報錯)。相當於是在TestClass類中又套了一個類,所以才會這樣寫。refreshLabMessage1當然就是testDelegate類構造方法的參數,用來指明委托哪個方法。最后把實例賦給同類型的mainThread。另外,在此例中mainThread委托了兩個方法,用+=運算符即可,如果想去除某個方法,亦可用-=運算符。
l 最后需要說明的就是跨線程訪問控件問題。窗體上的控件只允許創建它們的線程訪問,也就是主線程,如果非主線程訪問則會發生異常。我們可以借助於控件的InvokeRequired屬性來判斷該控件目前是否被主線程訪問,如果是,返回false。如果不是,再利用Invoke方法找到主線程,讓主線程執行訪問控件的方法,本例中借助於TestClass類中的mainThread對象,委托了訪問控件的方法refreshLabMessage1,再把mainThread對象傳入運行在主線程上的控件的Invoke方法即可。Invoke方法可以理解為:在哪個控件上調用了Invoke,就用那個控件所在的線程處理委托方法。在本例中用this調用Invoke方法,也就是窗體所在的線程,當然也是控件所在的線程。Invoke的兩個參數分別是:委托、委托的方法需要的參數。