亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > C# > 正文

C#中Timer使用及解決重入問題

2019-10-29 21:13:28
字體:
來源:轉載
供稿:網友

前言

打開久違的Live Writer,又已經好久沒寫博客了,真的太懶了。廢話不多說了,直接進入這次博客的主題--Timer。為什么要寫這個呢,因為前幾天應朋友之邀,想做個“黑客”小工具,功能挺簡單就是自動獲取剪貼板的內容然后發送郵件,就需要用到Timer來循環獲取剪貼板的內容,但是由于到了發送郵件這個功能,使用C#的SmtpClient始終發送不了郵件,以前寫過類似發郵件的功能,當時可以用網易的,現在也不能用了,不知道咋回事,只好作罷。在使用Timer中遇到了之前沒有想過的問題--重入。

介紹

首先簡單介紹一下timer,這里所說的timer是指的System.Timers.timer,顧名思義,就是可以在指定的間隔是引發事件。官方介紹在這里,摘抄如下:

Timer 組件是基于服務器的計時器,它使您能夠指定在應用程序中引發 Elapsed 事件的周期性間隔。然后可通過處理這個事件來提供常規處理。 例如,假設您有一臺關鍵性服務器,必須每周 7 天、每天 24 小時都保持運行。 可以創建一個使用 Timer 的服務,以定期檢查服務器并確保系統開啟并在運行。 如果系統不響應,則該服務可以嘗試重新啟動服務器或通知管理員。 基于服務器的 Timer 是為在多線程環境中用于輔助線程而設計的。 服務器計時器可以在線程間移動來處理引發的 Elapsed 事件,這樣就可以比 Windows 計時器更精確地按時引發事件。

如果想了解跟其他的timer有啥區別,可以看這里,里面有詳細的介紹,不再多說了(其實我也不知道還有這么多)。那使用這個計時器有啥好處呢?主要因為它是通過.NET Thread Pool實現的、輕量、計時精確、對應用程序及消息沒有特別的要求。

使用

下面就簡單介紹一下,這個Timer是怎么使用的,其實很簡單,我就采用微軟提供的示例來進行測試,直接上代碼了:

//Timer不要聲明成局部變量,否則會被GC回收 private static System.Timers.Timer aTimer; public static void Main() { //實例化Timer類,設置間隔時間為10000毫秒;  aTimer = new System.Timers.Timer(10000); //注冊計時器的事件 aTimer.Elapsed += new ElapsedEventHandler(OnTimedEvent); //設置時間間隔為2秒(2000毫秒),覆蓋構造函數設置的間隔 aTimer.Interval = 2000; //設置是執行一次(false)還是一直執行(true),默認為true aTimer.AutoReset = true; //開始計時 aTimer.Enabled = true; Console.WriteLine("按任意鍵退出程序。"); Console.ReadLine(); } //指定Timer觸發的事件 private static void OnTimedEvent(object source, ElapsedEventArgs e) { Console.WriteLine("觸發的事件發生在: {0}", e.SignalTime); }

運行的結果如下,計時蠻準確的:

/*按任意鍵退出程序。觸發的事件發生在: 2014/12/26 星期五 23:08:51觸發的事件發生在: 2014/12/26 星期五 23:08:53觸發的事件發生在: 2014/12/26 星期五 23:08:55觸發的事件發生在: 2014/12/26 星期五 23:08:57觸發的事件發生在: 2014/12/26 星期五 23:08:59*/

重入問題重現及分析

什么叫重入呢?這是一個有關多線程編程的概念:程序中,多個線程同時運行時,就可能發生同一個方法被多個進程同時調用的情況。當這個方法中存在一些非線程安全的代碼時,方法重入會導致數據不一致的情況。Timer方法重入是指使用多線程計時器,一個Timer處理還沒有完成,到了時間,另一Timer還會繼續進入該方法進行處理。下面演示一下重入問題的產生(可能重現的不是很好,不過也能簡單一下說明問題了):

//用來造成線程同步問題的靜態成員 private static int outPut = 1; //次數,timer沒調一次方法自增1 private static int num = 0; private static System.Timers.Timer timer = new System.Timers.Timer(); public static void Main() { timer.Interval = 1000; timer.Elapsed += TimersTimerHandler; timer.Start(); Console.WriteLine("按任意鍵退出程序。"); Console.ReadLine(); } /// <summary> /// System.Timers.Timer的回調方法 /// </summary> /// <param name="sender"></param> /// <param name="args"></param> private static void TimersTimerHandler(object sender, EventArgs args) { int t = ++num; Console.WriteLine(string.Format("線程{0}輸出:{1}, 輸出時間:{2}", t, outPut.ToString(),DateTime.Now)); System.Threading.Thread.Sleep(2000); outPut++; Console.WriteLine(string.Format("線程{0}自增1后輸出:{1},輸出時間:{2}", t, outPut.ToString(),DateTime.Now)); }

下面顯示一下輸出結果:

C#,timer

是不是感覺上面輸出結果很奇怪,首先是線程1輸出為1,沒有問題,然后隔了2秒后線程1自增1后輸出為2,這就有問題了,中間為什么還出現了線程2的輸出?更奇怪的是線程2剛開始輸出為1,自增1后盡然變成了3!其實這就是重入所導致的問題。別急,咱們分析一下就知道其中的緣由了。

首先timer啟動計時后,開啟一個線程1執行方法,當線程1第一次輸出之后,這時線程1休眠了2秒,此時timer并沒有閑著,因為設置的計時間隔為1秒,當在線程1休眠了1秒后,timer又開啟了線程2執行方法,線程2才不管線程1是執行中還是休眠狀態,所以此時線程2的輸出也為1,因為線程1還在休眠狀態,并沒有自增。然后又隔了1秒,這時發生同時發生兩個事件,線程1過了休眠狀態自增輸出為2,timer同時又開啟一個線程3,線程3輸出的為線程1自增后的值2,又過了1秒,線程2過了休眠狀態,之前的輸出已經是2,所以自增后輸出為3,又過了1秒……我都快暈了,大概就是這意思吧,我想表達的就是:一個Timer開啟的線程處理還沒有完成,到了時間,另一Timer還會繼續進入該方法進行處理。

那怎么解決這個問題呢?解決方案有三種,下面一一道來,適應不同的場景,不過還是推薦最后一種,比較安全。

重入問題解決方案

1、使用lock(Object)的方法來防止重入,表示一個Timer處理正在執行,下一個Timer發生的時候發現上一個沒有執行完就等待執行,適用重入很少出現的場景(具體也沒研究過,可能比較占內存吧)。

代碼跟上面差不多,在觸發的方法中加入lock,這樣當線程2進入觸發的方法中,發現已經被鎖,會等待鎖中的代碼處理完在執行,代碼如下:

private static object locko = new object();  /// <summary> /// System.Timers.Timer的回調方法 /// </summary> /// <param name="sender"></param> /// <param name="args"></param> private static void TimersTimerHandler(object sender, EventArgs args) { int t = ++num;  lock (locko) { Console.WriteLine(string.Format("線程{0}輸出:{1}, 輸出時間:{2}", t, outPut.ToString(), DateTime.Now)); System.Threading.Thread.Sleep(2000); outPut++; Console.WriteLine(string.Format("線程{0}自增1后輸出:{1},輸出時間:{2}", t, outPut.ToString(), DateTime.Now)); } }

 執行結果:

C#,timer

 2、設置一個標志,表示一個Timer處理正在執行,下一個Timer發生的時候發現上一個沒有執行完就放棄(注意這里是放棄,而不是等待哦,看看執行結果就明白啥意思了)執行,適用重入經常出現的場景。代碼如下:

 private static int inTimer = 0;  /// <summary> /// System.Timers.Timer的回調方法 /// </summary> /// <param name="sender"></param> /// <param name="args"></param> private static void TimersTimerHandler(object sender, EventArgs args) { int t = ++num; if (inTimer == 0) { inTimer = 1; Console.WriteLine(string.Format("線程{0}輸出:{1}, 輸出時間:{2}", t, outPut.ToString(), DateTime.Now)); System.Threading.Thread.Sleep(2000); outPut++; Console.WriteLine(string.Format("線程{0}自增1后輸出:{1},輸出時間:{2}", t, outPut.ToString(), DateTime.Now)); inTimer = 0; } }

執行結果:

C#,timer

3、在多線程下給inTimer賦值不夠安全,Interlocked.Exchange提供了一種輕量級的線程安全的給對象賦值的方法(感覺比較高上大,也是比較推薦的一種方法),執行結果與方法2一樣,也是放棄執行。Interlocked.Exchange用法參考這里。

private static int inTimer = 0;  /// <summary> /// System.Timers.Timer的回調方法 /// </summary> /// <param name="sender"></param> /// <param name="args"></param> private static void TimersTimerHandler(object sender, EventArgs args) { int t = ++num; if (Interlocked.Exchange(ref inTimer, 1) == 0) { Console.WriteLine(string.Format("線程{0}輸出:{1}, 輸出時間:{2}", t, outPut.ToString(), DateTime.Now)); System.Threading.Thread.Sleep(2000); outPut++; Console.WriteLine(string.Format("線程{0}自增1后輸出:{1},輸出時間:{2}", t, outPut.ToString(), DateTime.Now)); Interlocked.Exchange(ref inTimer, 0);  } }

C#,timer

執行結果:

總結

終于碼完字了,真心不容易啊。寫博客是個挺耗精力的事情,真心佩服那些大牛們筆耕不輟,致敬!在這里稍微總結一下,timer是一個使用挺簡單的類,拿來即用,這里主要總結了使用timer時重入問題的解決,以前也沒思考過這個問題,解決方案也挺簡單,在這里列出了三種,不知道還有沒有其他的方式。這里的解決方案同時也適用多線程的重入問題。

參考

在這里列出文章中沒有提及的參考,感謝各位前輩們智慧的結晶!

ASP.NET 定時器回調方法的重入

以上就是本文的全部內容,希望本文的內容對大家的學習或者工作能帶來一定的幫助,同時也希望多多支持VEVB武林網!


注:相關教程知識閱讀請移步到c#教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
久久全球大尺度高清视频| 久久精品视频中文字幕| 久久天天躁狠狠躁夜夜av| 精品香蕉在线观看视频一| 亚洲综合国产精品| 欧美丝袜一区二区三区| 黑人巨大精品欧美一区免费视频| 高清视频欧美一级| 91香蕉国产在线观看| 日韩在线观看网址| 中日韩美女免费视频网站在线观看| 国产视频自拍一区| 欧美电影在线观看网站| 久久久久久免费精品| 成人免费福利在线| 亚洲免费小视频| 91九色在线视频| 136fldh精品导航福利| 久久国产精品影片| 国产女同一区二区| 北条麻妃一区二区三区中文字幕| 北条麻妃在线一区二区| 日韩精品免费一线在线观看| 少妇激情综合网| 国产一区二区av| 国产精品亚洲综合天堂夜夜| 精品国产乱码久久久久酒店| 久久综合免费视频影院| 欧美中文在线观看国产| 久久综合色影院| 成人福利网站在线观看| 日韩av免费网站| 精品国产精品自拍| 精品亚洲男同gayvideo网站| 精品福利在线看| 欧美性xxxx18| 在线播放国产一区中文字幕剧情欧美| 91精品国产777在线观看| 国产亚洲成av人片在线观看桃| 欧美日韩一区二区三区| 国产亚洲一区二区精品| 亚洲精品永久免费| 午夜免费在线观看精品视频| 久久久人成影片一区二区三区| 91中文字幕在线观看| 亚洲国产精品久久久久秋霞蜜臀| 精品二区三区线观看| 国产精品视频999| 久久久久久久成人| 国产一区二区香蕉| 国产精品亚洲综合天堂夜夜| 国产精品国语对白| 久久免费观看视频| 欧美极品美女视频网站在线观看免费| 97在线精品国自产拍中文| 国产精品av在线| www欧美日韩| 国外成人免费在线播放| 欧美裸体男粗大视频在线观看| 午夜精品久久久久久久久久久久| 91精品国产自产91精品| 欧美放荡办公室videos4k| 国产日韩欧美视频在线| 国产成人免费av电影| 亚洲福利精品在线| 国产精品美女无圣光视频| 成人免费直播live| 日韩男女性生活视频| 国产网站欧美日韩免费精品在线观看| 日韩电影中文字幕一区| 欧美久久精品一级黑人c片| 久久久久久香蕉网| 日韩欧美国产成人| 亚洲国产小视频在线观看| 日韩高清人体午夜| 亚洲第一中文字幕| 欧美中文字幕在线播放| 日韩视频―中文字幕| 国产伦精品一区二区三区精品视频| 亚洲美女动态图120秒| 国模私拍视频一区| 91色琪琪电影亚洲精品久久| 91色在线观看| 国产精品亚洲自拍| 久久久噜噜噜久久久| 亚洲性av网站| 日韩在线视频中文字幕| 久久久久久亚洲精品| 日韩av在线电影网| 久久精品国产视频| 国产精品十八以下禁看| 欧美黑人性猛交| 97成人超碰免| 日韩电影大全免费观看2023年上| 欧美中文字幕精品| 亚洲www永久成人夜色| 久久久精品久久久久| 日本在线精品视频| 国产精品高清免费在线观看| 色综合天天综合网国产成人网| 在线播放日韩精品| 成人久久一区二区三区| 亚洲欧美日韩国产中文| 日韩免费黄色av| 亚洲三级免费看| 精品久久久久久久久中文字幕| 国产精品久久久久高潮| 黑人巨大精品欧美一区二区三区| 精品香蕉在线观看视频一| 亚洲精品综合久久中文字幕| 亚洲视频一区二区| 欧美精品videofree1080p| 国产欧美在线看| 久久色在线播放| 亚洲最大的成人网| 大荫蒂欧美视频另类xxxx| 成人激情av在线| 国产丝袜一区二区| 国产精品久久久久久久天堂| 国产精品男人的天堂| 一个人看的www久久| 久久久久久久爱| 成人午夜在线视频一区| 欧美美女操人视频| 国产免费一区二区三区香蕉精| 日韩在线视频观看| 国产精品免费福利| 一区二区三区美女xx视频| 久久99亚洲精品| 亚洲四色影视在线观看| 久久久久亚洲精品| 亚洲欧美精品一区二区| 日韩精品在线视频观看| 欧美成人免费在线观看| 亚洲人a成www在线影院| 日韩在线观看免费全| 欧美激情网站在线观看| 亚洲aⅴ男人的天堂在线观看| 国产偷国产偷亚洲清高网站| 国产精品久久久久av免费| 久久久女女女女999久久| 成人午夜两性视频| 亚洲无限乱码一二三四麻| 亚洲精品网址在线观看| 欧美色另类天堂2015| 狠狠躁夜夜躁久久躁别揉| 2021久久精品国产99国产精品| 欧美极品在线视频| 久久躁日日躁aaaaxxxx| 国产网站欧美日韩免费精品在线观看| 欧美激情二区三区| 亚洲精品电影在线观看| 精品亚洲aⅴ在线观看| 亚洲黄色片网站| 亚洲国产欧美在线成人app| 亚洲欧美精品一区| 911国产网站尤物在线观看| 国产丝袜精品第一页| 日韩在线观看电影| 欧美丝袜一区二区| 97视频国产在线| 国产日产欧美a一级在线| 俺去啦;欧美日韩| 久久99精品国产99久久6尤物|