C#中提供了三種類型的計時器:

1
、基於 Windows 的標準計時器(System.Windows.Forms.Timer)

2
、基於伺服器的計時器(System.Timers.Timer)

3
、線程計時器(System.Threading.Timer)

下面我就通過一些小實驗來具體分析三種計時器使用上面的異同點,特別是和線程有關的部分。

實驗例子截圖:



一、基於 Windows 的標準計時器(System.Windows.Forms.Timer)

首先注意一點就是:Windows 計時器是為單線程環境設計的

此計時器從Visual Basic 1.0 版起就存在於該產品中,並且基本上未做改動

這個計時器是使用最簡單的一種,只要把工具箱中的Timer控制項拖到表單上,然後設置一下事件和間隔時間等屬性就可以了



實驗出來的結果也完全符合單線程的特點:

1
、當啟動此計時器後,會在下方子線程ID列表中顯示子線程ID,並且和主線程ID相同

 



        private void formsTimer_Tick(object sender, EventArgs e)

        
{

            i++;

            lblSubThread.Text += 
"
子線程執行,線程ID" + System.Threading.Thread.CurrentThread.ManagedThreadId.ToString() + "\r\n";

        }


 





2
、當單擊主線程暫停5秒後,子線程會暫停執行,並且當5秒之後不會執行之前被暫停的子線程,而是直接執行後面的子線程(也就是會少輸出幾行值)

            System.Threading.Thread.Sleep(5000);

3
、在子進程的事件中暫停5秒會導致主視窗相應無回應5

4
、定義一個線程靜態變數:

 
       [ThreadStatic]

        private static int i = 0;

在子線程事件中每次加一,再點擊線程靜態變數值會得到增加後的i



二、基於伺服器的計時器(System.Timers.Timer)

System.Timers.Timer
不依賴表單,是從線程池喚醒線程,是傳統的計時器為了在伺服器環境上運行而優化後的更新版本

VS2005的工具箱中沒有提供現成的控制項,需要手工編碼使用此計時器

使用方式有兩種,

1
、通過SynchronizingObject屬性依附於表單

 



            System.Timers.Timer timersTimer = new System.Timers.Timer();

            

            timersTimer.Enabled = 
false;

            timersTimer.Interval = 
100;

            timersTimer.Elapsed += 
new System.Timers.ElapsedEventHandler(timersTimer_Elapsed);



            timersTimer.SynchronizingObject = 
this;


 



通過這種方式來使用,實驗效果幾乎和基於 Windows 的標準計時器一樣,只是在上面的第二條實驗中,雖然也會暫停子線程的執行,不過在5秒之後把之前排隊的任務都執行掉(也就是不會少輸出幾行值)

2
、不使用SynchronizingObject屬性

這種方式就是多線程的方式了,即啟動的子線程和主表單不在一個線程。不過這樣也存在一個問題:由於子線程是單獨的一個線程,那麼就不能訪問住表單中的控制項了,只能通過代理的方式來訪問:

 



        delegate void SetTextCallback(string text);

         .

         .

        
void timersTimer_Elapsed(object sender, System.Timers.ElapsedEventArgs e)

        
{

            
//
使用代理

            string text = "子線程執行,線程ID" + System.Threading.Thread.CurrentThread.ManagedThreadId.ToString() + "\r\n";

            SetTextCallback d = 
new SetTextCallback(SetText);

            
this.Invoke(d, new object[]  { text });

            i++;

        }



        
private void SetText(string text)

        
{

            lblSubThread.Text += text;

        }

 



這樣我們再次實驗就會得到如下的結果:

1
、當啟動此計時器後,會在下方子線程ID列表中顯示子線程ID,並且和主線程ID不相同



2
、當單擊主線程暫停5秒後,子線程會一直往下執行(介面上可能看不出來,不過通過在子線程輸出檔的方式可以很方便的看出來)

3
、在子進程的事件中暫停5秒不會導致主視窗無回應

4
、在子線程事件中每次給線程靜態變數加一,再點擊線程靜態變數值得到的值還是0(不會改變主視窗中的線程靜態變數)  

arrow
arrow
    全站熱搜

    狼翔月影 發表在 痞客邦 留言(0) 人氣()