三分鐘解決VB.NET定時(shí)器事件重入
喜歡VB.NET朋友一定會(huì)很喜歡研究定時(shí)器方面的問(wèn)題吧。本人就很喜歡,在網(wǎng)上我收集了很多東西,在這個(gè)就VB.NET定時(shí)器事件重入的問(wèn)題來(lái)和大家一起探討一下。不論在客戶端應(yīng)用程序還是服務(wù)器組件(包括窗口服務(wù))定時(shí)器通常扮演一個(gè)重要的角色。寫(xiě)一個(gè)高效的定時(shí)器驅(qū)動(dòng)型可管理代碼要求對(duì)程序流程有一個(gè)清晰的理解及掌握.NET線程模型的精妙之處。.NET框架類庫(kù)提供了三種不同的定時(shí)器類:System.Windows.Forms.Timer, System.Timers.Timer, 和System.Threading.Timer。每個(gè)類為不同的場(chǎng)合進(jìn)行設(shè)計(jì)和優(yōu)化。本文章將研究這三個(gè)類并讓你理解如何及何時(shí)應(yīng)該使用哪一個(gè)類。
VB.NET定時(shí)器事件處理重入
當(dāng)和異步定時(shí)器事件打交道時(shí),如由System.Timers.Timer和System.Threading.Timer產(chǎn)生的定時(shí)器事件,有另外一個(gè)細(xì)微之處你需要考慮。問(wèn)題就是必須處理代碼重入。如果你的定時(shí)器事件處理函數(shù)代碼執(zhí)行時(shí)間比你的定時(shí)器引發(fā)定時(shí)器事件的時(shí)間間隔要長(zhǎng),你預(yù)先又沒(méi)有采取必要的措施保護(hù)防止多線程訪問(wèn)你的對(duì)象和變量,你就會(huì)陷入調(diào)試的困境。看一下下面的代碼片斷:
- private int tickCounter = 0;
- private void tmrTimersTimer_Elapsed(object sender, System.Timers.ElapsedEventArgse)
- {
- System.Threading.Interlocked.Increment(ref tickCounter);
- Thread.Sleep(5000);
- MessageBox.Show(tickCounter.ToString());
- }
假設(shè)你的定時(shí)器間隔屬性設(shè)置為1000毫秒,你也許會(huì)奇怪當(dāng)***個(gè)信息框彈出時(shí)顯示的值是5。這是因?yàn)樵谶@5秒期間***個(gè)定時(shí)器事件正在睡眠,而定時(shí)器卻在不同的工作者線程上繼續(xù)產(chǎn)生時(shí)間消失事件。因此,在***個(gè)VB.NET定時(shí)器事件處理完成之前tickCounter變量被增加了5次。注意我使用了Interlocked.Increment方法以線程安全的方式增加tickCounter變量的值。也有其它方法可以這樣做,但是Interlock.Increment是為這種操作而特別設(shè)計(jì)的。
解決這種問(wèn)題的簡(jiǎn)單方法就是在你的事件處理函數(shù)代碼塊中暫時(shí)禁止定時(shí)器,接著再允許定時(shí)器,就像下面的代碼:
- private void tmrTimersTimer_Elapsed(object sender, System.Timers.ElapsedEventArgse)
- {
- tmrTimers.Enabled = false;
- System.Threading.Interlocked.Increment(ref tickCounter);
- Thread.Sleep(5000);
- MessageBox.Show(tickCounter.ToString());
- tmrTimersTimer.Enabled = true;
- }
有了這段代碼,消息框就會(huì)每5秒鐘顯示一次,就像你所期望的那樣,tickCounter的值每次只增加1。另外一些可選的原始同步對(duì)象就是Monitor或mutex去確保所有將來(lái)的事件被排隊(duì)直到當(dāng)前的事件處理函數(shù)執(zhí)行完成。
結(jié)論
當(dāng)使用定時(shí)器類時(shí)有一點(diǎn)你要考慮的就是是否可以使用Windows調(diào)度器去定期的運(yùn)行標(biāo)準(zhǔn)的可執(zhí)行程序來(lái)更簡(jiǎn)單的解決問(wèn)題。
【編輯推薦】