C#利用委托跨線程更新UI數據


         在使用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的兩個參數分別是:委托、委托的方法需要的參數。

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM