C中三種定時(shí)器對象的比較_第1頁
C中三種定時(shí)器對象的比較_第2頁
C中三種定時(shí)器對象的比較_第3頁
C中三種定時(shí)器對象的比較_第4頁
C中三種定時(shí)器對象的比較_第5頁
已閱讀5頁,還剩5頁未讀 繼續(xù)免費(fèi)閱讀

下載本文檔

版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請進(jìn)行舉報(bào)或認(rèn)領(lǐng)

文檔簡介

1、C#中三種定時(shí)器對象的比較關(guān)于C#中timer類 在C#里關(guān)于定時(shí)器類就有3個(gè)1.定義在System.Windows.Forms里2.定義在System.Threading.Timer類里3.定義在System.Timers.Timer類里System.Windows.Forms.Timer是應(yīng)用于WinForm中的,它是通過Windows消息機(jī)制實(shí)現(xiàn)的,類似于VB或Delphi中的Timer控件,內(nèi)部使用API SetTimer實(shí)現(xiàn)的。它的主要缺點(diǎn)是計(jì)時(shí)不精確,而且必須有消息循環(huán),Console Application(控制臺(tái)應(yīng)用程序)無法使用。System.Timers.Timer和Sys

2、tem.Threading.Timer非常類似,它們是通過.NET Thread Pool實(shí)現(xiàn)的,輕量,計(jì)時(shí)精確,對應(yīng)用程序、消息沒有特別的要求。System.Timers.Timer還可以應(yīng)用于WinForm,完全取代上面的Timer控件。它們的缺點(diǎn)是不支持直接的拖放,需要手工編碼。例:使用System.Timers.Timer類System.Timers.Timer t = new System.Timers.Timer(10000);/實(shí)例化Timer類,設(shè)置間隔時(shí)間為10000毫秒;t.Elapsed += new System.Timers.ElapsedEventHandler(t

3、heout);/到達(dá)時(shí)間的時(shí)候執(zhí)行事件;t.AutoReset = true;/設(shè)置是執(zhí)行一次(false)還是一直執(zhí)行(true);t.Enabled = true;/是否執(zhí)行System.Timers.Timer.Elapsed事件;public void theout(object source, System.Timers.ElapsedEventArgs e)MessageBox.Show(OK!);實(shí)驗(yàn)分析C#中三種計(jì)時(shí)器使用異同點(diǎn)C#中提供了三種類型的計(jì)時(shí)器:1、基于 Windows 的標(biāo)準(zhǔn)計(jì)時(shí)器(System.Windows.Forms.Timer)2、基于服務(wù)器的計(jì)時(shí)器(Sy

4、stem.Timers.Timer)3、線程計(jì)時(shí)器(System.Threading.Timer)下面我就通過一些小實(shí)驗(yàn)來具體分析三種計(jì)時(shí)器使用上面的異同點(diǎn),特別是和線程有關(guān)的部分。實(shí)驗(yàn)例子截圖:一、基于 Windows 的標(biāo)準(zhǔn)計(jì)時(shí)器(System.Windows.Forms.Timer)首先注意一點(diǎn)就是:Windows 計(jì)時(shí)器是為單線程環(huán)境設(shè)計(jì)的此計(jì)時(shí)器從Visual Basic 1.0 版起就存在于該產(chǎn)品中,并且基本上未做改動(dòng)這個(gè)計(jì)時(shí)器是使用最簡單的一種,只要把工具箱中的Timer控件拖到窗體上,然后設(shè)置一下事件和間隔時(shí)間等屬性就可以了實(shí)驗(yàn)出來的結(jié)果也完全符合單線程的特點(diǎn):1、當(dāng)啟動(dòng)此計(jì)時(shí)

5、器后,會(huì)在下方子線程ID列表中顯示子線程ID,并且和主線程ID相同private void formsTimer_Tick(object sender, EventArgs e)i+;lblSubThread.Text += 子線程執(zhí)行,線程ID: + System.Threading.Thread.CurrentThread.ManagedThreadId.ToString() + rn;2、當(dāng)單擊主線程暫停5秒后,子線程會(huì)暫停執(zhí)行,并且當(dāng)5秒之后不會(huì)執(zhí)行之前被暫停的子線程,而是直接執(zhí)行后面的子線程(也就是會(huì)少輸出幾行值)System.Threading.Thread.Sleep(5000)

6、;3、在子進(jìn)程的事件中暫停5秒會(huì)導(dǎo)致主窗口相應(yīng)無響應(yīng)5秒4、定義一個(gè)線程靜態(tài)變量:ThreadStaticprivate static int i = 0;在子線程事件中每次加一,再點(diǎn)擊線程靜態(tài)變量值會(huì)得到增加后的i值二、基于服務(wù)器的計(jì)時(shí)器(System.Timers.Timer)System.Timers.Timer不依賴窗體,是從線程池喚醒線程,是傳統(tǒng)的計(jì)時(shí)器為了在服務(wù)器環(huán)境上運(yùn)行而優(yōu)化后的更新版本在VS2005的工具箱中沒有提供現(xiàn)成的控件,需要手工編碼使用此計(jì)時(shí)器使用方式有兩種,1、通過SynchronizingObject屬性依附于窗體System.Timers.Timer timer

7、sTimer = new System.Timers.Timer();timersTimer.Enabled = false;timersTimer.Interval = 100;timersTimer.Elapsed += new System.Timers.ElapsedEventHandler(timersTimer_Elapsed);timersTimer.SynchronizingObject = this;通過這種方式來使用,實(shí)驗(yàn)效果幾乎和基于 Windows 的標(biāo)準(zhǔn)計(jì)時(shí)器一樣,只是在上面的第二條實(shí)驗(yàn)中,雖然也會(huì)暫停子線程的執(zhí)行,不過在5秒之后把之前排隊(duì)的任務(wù)都執(zhí)行掉(也就是不會(huì)少

8、輸出幾行值)2、不使用SynchronizingObject屬性這種方式就是多線程的方式了,即啟動(dòng)的子線程和主窗體不在一個(gè)線程。不過這樣也存在一個(gè)問題:由于子線程是單獨(dú)的一個(gè)線程,那么就不能訪問住窗體中的控件了,只能通過代理的方式來訪問:delegate void SetTextCallback(string text);。void timersTimer_Elapsed(object sender, System.Timers.ElapsedEventArgs e)/使用代理string text = 子線程執(zhí)行,線程ID: + System.Threading.Thread.Current

9、Thread.ManagedThreadId.ToString() + rn;SetTextCallback d = new SetTextCallback(SetText);this.Invoke(d, new object text );i+;private void SetText(string text)lblSubThread.Text += text;這樣我們再次實(shí)驗(yàn)就會(huì)得到如下的結(jié)果:1、當(dāng)啟動(dòng)此計(jì)時(shí)器后,會(huì)在下方子線程ID列表中顯示子線程ID,并且和主線程ID不相同2、當(dāng)單擊主線程暫停5秒后,子線程會(huì)一直往下執(zhí)行(界面上可能看不出來,不過通過在子線程輸出文件的方式可以很方便的看

10、出來)3、在子進(jìn)程的事件中暫停5秒不會(huì)導(dǎo)致主窗口無響應(yīng)4、在子線程事件中每次給線程靜態(tài)變量加一,再點(diǎn)擊線程靜態(tài)變量值得到的值還是0(不會(huì)改變主窗口中的線程靜態(tài)變量)三、線程計(jì)時(shí)器(System.Threading.Timer)線程計(jì)時(shí)器也不依賴窗體,是一種簡單的、輕量級(jí)計(jì)時(shí)器,它使用回調(diào)方法而不是使用事件,并由線程池線程提供支持。對消息不在線程上發(fā)送的方案中,線程計(jì)時(shí)器是非常有用的。使用方法如下:System.Threading.Timer threadTimer;public void ThreadMethod(Object state)/使用代理string text = 子線程執(zhí)行,線程

11、ID: + System.Threading.Thread.CurrentThread.ManagedThreadId.ToString() + rn;SetTextCallback d = new SetTextCallback(SetText);this.Invoke(d, new object text );i+;private void Form1_Load(object sender, EventArgs e)threadTimer = new System.Threading.Timer(new System.Threading.TimerCallback(ThreadMethod

12、), null, -1, -1);暫停代碼:threadTimer.Change(-1, -1);實(shí)驗(yàn)的效果和基于服務(wù)器的計(jì)時(shí)器(System.Timers.Timer)的第二種方式是一樣的,當(dāng)然具體的使用方法和原理是不一樣的,最主要的就是這種方式使用的是代理的方式而不是事件的方式,并且可以不依賴于窗體和組件而單獨(dú)執(zhí)行C# 各種定時(shí)器比較(轉(zhuǎn)載) 1.單線程的定時(shí)器存在很多問題,定時(shí)器只是定時(shí)把消息WM_TIMER訪到線程的消息隊(duì)列里,但并不保證消息會(huì)立即被響應(yīng),如果碰巧系統(tǒng)比較忙,消息可能會(huì)在隊(duì)列里放一段時(shí)間才被響應(yīng),這樣會(huì)造成本來應(yīng)該間隔一段時(shí)間發(fā)生的消息連續(xù)發(fā)生了。2. 。NET Fra

13、meWork帶來了新的解決方案Server Timers基于服務(wù)器的計(jì)時(shí)器,位于工具箱的“組件”選項(xiàng)卡上Thread Timers在編程時(shí)使用的線程計(jì)時(shí)器Windows Timers基于 Windows 的標(biāo)準(zhǔn)計(jì)時(shí)器,工具箱的Windows 窗體選項(xiàng)卡上; 區(qū)別aWindows Timers 提供了和WinAPI一樣的Timer 基于消息,仍然是單線程其他兩個(gè)是是基于線程池的Thread Pool【最大好處,產(chǎn)生的時(shí)間準(zhǔn)確均勻】 區(qū)別bServer Timers 和 Thread Timers 的不同在于ServerTimers 是基于事件的,Thread Timers是基于回調(diào)函數(shù) Thre

14、ad Timer是一個(gè)輕量級(jí)的方便使用,但也要注意一些問題,由于是多線程定時(shí)器,就會(huì)出現(xiàn)如果一個(gè)Timer處理沒有完成,到了時(shí)間下一個(gè)照樣會(huì)發(fā)生,導(dǎo)致嚴(yán)重錯(cuò)誤,對付重入問題,通常的辦法是加鎖,但對于Timer不能簡單的這樣處理。使用Timer來處于的事情,要注意:首先Timer處理里本來就不應(yīng)該做太需要時(shí)間的事情,或者花費(fèi)時(shí)間無法估計(jì)的事情,比同遠(yuǎn)方的服務(wù)器建立一個(gè)網(wǎng)絡(luò)連接,這樣的做法盡量避免如果實(shí)在無法避免,那么要評(píng)估Timer處理超時(shí)是否經(jīng)常發(fā)生,如果是很少出現(xiàn),那么可以用lock(Object)的方法來防止重入如果這種情況經(jīng)常出現(xiàn)呢?那就要用另外的方法來防止重入了我們可以設(shè)置一個(gè)標(biāo)志,表

15、示一個(gè)Timer處理正在執(zhí)行,下一個(gè)Timer發(fā)生的時(shí)候發(fā)現(xiàn)上一個(gè)沒有執(zhí)行完就放棄執(zhí)行static int inTimer = 0;public static void threadTimerCallback(Object obj)if ( inTiemr = 0 )inTimer = 1; Console.WriteLine(Time:0, tThread ID:1, DateTime.Now, Thread.CurrentThread.GetHashCode();Thread.Sleep(2000);inTimer = 0;但是在多線程下給inTimer賦值不夠安全,還好Interlock

16、ed.Exchange提供了一種輕量級(jí)的線程安全的給對象賦值的方法static int inTimer = 0;public static void threadTimerCallback(Object obj)if ( Interlocked.Exchange(ref inTimer, 1) = 0 )Console.WriteLine(Time:0, tThread ID:1, DateTime.Now, Thread.CurrentThread.GetHashCode();Thread.Sleep(250);Interlocked.Exchange(ref inTimer, 0);=試驗(yàn)

17、=1.ThreadTimer = new System.Threading.Timer(new TimerCallBack(onTime),this,0,1000)- 參數(shù) -a.回調(diào)方法,b.回調(diào)方法中使用信息的對象,c.表示在 callback 參數(shù)調(diào)用它的方法之前延遲的時(shí)間量。指定 -1 毫秒以防止啟動(dòng)計(jì)時(shí)器。指定零 (0) 以立即啟動(dòng)計(jì)時(shí)器。d.在調(diào)用 callback 所引用的方法之間的時(shí)間間隔。指定 -1 毫秒可以禁用定期終止2.BaseTimeHandler 自定義一個(gè)事件void OnTime(Object State)Control ctl = State as Form1;if ( ctl != null & BaseTimeHandler != null)if ( ctl.IsHandleCreated )DateTime dt = DateTi

溫馨提示

  • 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
  • 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
  • 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會(huì)有圖紙預(yù)覽,若沒有圖紙預(yù)覽就沒有圖紙。
  • 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
  • 5. 人人文庫網(wǎng)僅提供信息存儲(chǔ)空間,僅對用戶上傳內(nèi)容的表現(xiàn)方式做保護(hù)處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負(fù)責(zé)。
  • 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請與我們聯(lián)系,我們立即糾正。
  • 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時(shí)也不承擔(dān)用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。

最新文檔

評(píng)論

0/150

提交評(píng)論