欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

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

 更新時間:2017年02月20日 09:15:18   作者:劉來來  
本文主要介紹了C#中Timer使用及解決重入問題的相關(guān)知識。具有很好的參考價值,下面跟著小編一起來看下吧

前言

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

介紹

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

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

如果想了解跟其他的timer有啥區(qū)別,可以看這里,里面有詳細(xì)的介紹,不再多說了(其實(shí)我也不知道還有這么多)。那使用這個計(jì)時器有啥好處呢?主要因?yàn)樗峭ㄟ^.NET Thread Pool實(shí)現(xiàn)的、輕量、計(jì)時精確、對應(yīng)用程序及消息沒有特別的要求。

使用

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

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

運(yùn)行的結(jié)果如下,計(jì)時蠻準(zhǔn)確的:

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

重入問題重現(xiàn)及分析

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

//用來造成線程同步問題的靜態(tài)成員
 private static int outPut = 1;
 //次數(shù),timer沒調(diào)一次方法自增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的回調(diào)方法
 /// </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));
 }

下面顯示一下輸出結(jié)果:

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

首先timer啟動計(jì)時后,開啟一個線程1執(zhí)行方法,當(dāng)線程1第一次輸出之后,這時線程1休眠了2秒,此時timer并沒有閑著,因?yàn)樵O(shè)置的計(jì)時間隔為1秒,當(dāng)在線程1休眠了1秒后,timer又開啟了線程2執(zhí)行方法,線程2才不管線程1是執(zhí)行中還是休眠狀態(tài),所以此時線程2的輸出也為1,因?yàn)榫€程1還在休眠狀態(tài),并沒有自增。然后又隔了1秒,這時發(fā)生同時發(fā)生兩個事件,線程1過了休眠狀態(tài)自增輸出為2,timer同時又開啟一個線程3,線程3輸出的為線程1自增后的值2,又過了1秒,線程2過了休眠狀態(tài),之前的輸出已經(jīng)是2,所以自增后輸出為3,又過了1秒……我都快暈了,大概就是這意思吧,我想表達(dá)的就是:一個Timer開啟的線程處理還沒有完成,到了時間,另一Timer還會繼續(xù)進(jìn)入該方法進(jìn)行處理。

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

重入問題解決方案

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

代碼跟上面差不多,在觸發(fā)的方法中加入lock,這樣當(dāng)線程2進(jìn)入觸發(fā)的方法中,發(fā)現(xiàn)已經(jīng)被鎖,會等待鎖中的代碼處理完在執(zhí)行,代碼如下:

private static object locko = new object(); 
 /// <summary>
 /// System.Timers.Timer的回調(diào)方法
 /// </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)); } }

 執(zhí)行結(jié)果:

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

 private static int inTimer = 0; 
 /// <summary>
 /// System.Timers.Timer的回調(diào)方法
 /// </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;
 }
 }

執(zhí)行結(jié)果:

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

private static int inTimer = 0; 
 /// <summary>
 /// System.Timers.Timer的回調(diào)方法
 /// </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); 
 }
 }

執(zhí)行結(jié)果:

總結(jié)

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

參考

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

ASP.NET 定時器回調(diào)方法的重入

以上就是本文的全部內(nèi)容,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作能帶來一定的幫助,同時也希望多多支持腳本之家!

相關(guān)文章

  • C#中各種計(jì)時器用法小結(jié)

    C#中各種計(jì)時器用法小結(jié)

    這篇文章主要介紹了C#中各種計(jì)時器用法,結(jié)合實(shí)例形式總結(jié)分析了C#中各種常用時間相關(guān)類實(shí)現(xiàn)計(jì)時器功能的操作技巧,需要的朋友可以參考下
    2017-06-06
  • C#之字符串截取--Regex.Match使用

    C#之字符串截取--Regex.Match使用

    這篇文章主要介紹了C#之字符串截取--Regex.Match使用解讀,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2023-05-05
  • C#編程實(shí)現(xiàn)QQ界面的方法

    C#編程實(shí)現(xiàn)QQ界面的方法

    這篇文章主要介紹了C#編程實(shí)現(xiàn)QQ界面的方法,較為詳細(xì)的分析了C#實(shí)現(xiàn)模擬QQ登錄界面的具體步驟與相關(guān)實(shí)現(xiàn)代碼,需要的朋友可以參考下
    2016-03-03
  • Unity游戲開發(fā)中必備的設(shè)計(jì)模式之外觀模式詳解

    Unity游戲開發(fā)中必備的設(shè)計(jì)模式之外觀模式詳解

    外觀模式是一種結(jié)構(gòu)型設(shè)計(jì)模式,為復(fù)雜系統(tǒng)提供了簡單的接口,使得子系統(tǒng)間的通信更加簡潔和易于維護(hù)。在Unity游戲開發(fā)中,外觀模式可以幫助開發(fā)者更好地管理游戲?qū)ο蠛徒M件等復(fù)雜結(jié)構(gòu)
    2023-05-05
  • 詳解WPF如何動態(tài)生成DataGrid的行和列

    詳解WPF如何動態(tài)生成DataGrid的行和列

    在日常開發(fā)中,DataGrid作為二維表格,非常適合數(shù)據(jù)的展示和統(tǒng)計(jì),本文以一些簡單的小例子,簡述在WPF開發(fā)中,如何動態(tài)生成DataGrid的行和列,需要的可以了解下
    2024-02-02
  • C#中的WebRequest與WebResponse抽象類、DNS靜態(tài)類、Ping類介紹

    C#中的WebRequest與WebResponse抽象類、DNS靜態(tài)類、Ping類介紹

    這篇文章介紹了C#中的WebRequest與WebResponse抽象類、DNS靜態(tài)類、Ping類,文中通過示例代碼介紹的非常詳細(xì)。對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2022-05-05
  • C#實(shí)現(xiàn)DevExpress本地化實(shí)例詳解

    C#實(shí)現(xiàn)DevExpress本地化實(shí)例詳解

    這篇文章主要介紹了C#實(shí)現(xiàn)DevExpress本地化,以實(shí)例形式較為詳細(xì)的分析了DevExpress本地化的相關(guān)技巧,具有一定參考借鑒價值,需要的朋友可以參考下
    2015-08-08
  • C#簡單嵌套flash讀取數(shù)據(jù)的實(shí)現(xiàn)代碼

    C#簡單嵌套flash讀取數(shù)據(jù)的實(shí)現(xiàn)代碼

    這篇文章主要介紹了C#簡單嵌套flash讀取數(shù)據(jù)的實(shí)現(xiàn)代碼,有需要的朋友可以參考一下
    2013-11-11
  • C#實(shí)現(xiàn)自定義打印文字和圖片的示例代碼

    C#實(shí)現(xiàn)自定義打印文字和圖片的示例代碼

    本文主要介紹了C#實(shí)現(xiàn)自定義打印文字和圖片的示例代碼,C#中打印其實(shí)就是自己繪圖+調(diào)用系統(tǒng)打印函數(shù),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2022-04-04
  • C# 使用Fiddler捕獲本地HttpClient發(fā)出的請求操作

    C# 使用Fiddler捕獲本地HttpClient發(fā)出的請求操作

    這篇文章主要介紹了C# 使用Fiddler捕獲本地HttpClient發(fā)出的請求操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-10-10

最新評論