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

詳解C#中的定時器Timer類及其垃圾回收機制

 更新時間:2016年04月27日 10:32:37   作者:go,go  
這篇文章主要介紹了C#中的定時器Timer類及其垃圾回收機制,講解了Timer相關的單線程異步工作,需要的朋友可以參考下

關于C# Timer類  在C#里關于定時器類就有3個

C# Timer使用的方法1.定義在System.Windows.Forms里

C# Timer使用的方法2.定義在System.Threading.Timer類里  "

C# Timer使用的方法3.定義在System.Timers.Timer類里

下面我們來具體看看這3種C# Timer用法的解釋:

(1)System.Windows.Forms.Timer

應用于WinForm中的,它是通過Windows消息機制實現(xiàn)的,類似于VB或Delphi中的Timer控件,內(nèi)部使用API  SetTimer實現(xiàn)的。它的主要缺點是計時不精確,而且必須有消息循環(huán),Console  Application(控制臺應用程序)無法使用。  
 
(2)System.Timers.Timer

和System.Threading.Timer非常類似,它們是通過.NET  Thread  Pool實現(xiàn)的,輕量,計時精確,對應用程序、消息沒有特別的要求。

(3)System.Timers.Timer還可以應用于WinForm,完全取代上面的Timer控件。它們的缺點是不支持直接的拖放,需要手工編碼。

C# Timer用法實例

使用System.Timers.Timer類

System.Timers.Timer t =  
new System.Timers.Timer(10000); 
//實例化Timer類,設置間隔時間為10000毫秒;  
t.Elapsed +=  
new System.Timers.ElapsedEventHandler(theout); 
//到達時間的時候執(zhí)行事件;  
t.AutoReset = true; 
//設置是執(zhí)行一次(false)還是一直執(zhí)行(true);  
t.Enabled = true; 
//是否執(zhí)行System.Timers.Timer.Elapsed事件;  
 
public void theout( 
object source,  
System.Timers.ElapsedEventArgs e)  
 {  
  MessageBox.Show("OK!");  
 } 

 
Timer的垃圾回收機制
通常我們需要定時執(zhí)行一段任務的時候,我們就需要定時器,這時我們就可以使用c# System.Threading空間中的 Timer定時器;他是個異步定時器,時間到時每次都是在線程池中分配一個線程去執(zhí)行任務。下面我們來看一個有趣的例子:

class Program
  {
    static void Main(string[] args)
    {
      Timer timer = new Timer(TimerCallback,null,0,2000);
      
      Console.ReadLine();
    }
 
    private static void TimerCallback(object o)
    {
      Console.WriteLine("in TimerCallback method");
      GC.Collect();
 
      
    }
  }

當我們在debug模式下運行該段程序時,正如我們期盼的那樣程序會每隔2秒鐘執(zhí)行該方法,打印出"in TimerCallback method”,而在release模式下執(zhí)行的時候,只執(zhí)行一次該方法,字符串只打印一次。在這里我們在調(diào)用TimerCallback方法時,強制執(zhí)行垃圾回收器,說明在release模式下,垃圾回收器執(zhí)行回收算法時,首先假設所有對象都是可回收的,當將Timer對象賦值給變量t后,t沒有在被引用,因此也就沒有變量引用Timer對象,所以垃圾收集這時會回收Timer對象。那么為什么在debug模式下卻能夠運行能,這跟c#編譯器的優(yōu)化方式有關,在release模式下編譯器做了相關的優(yōu)化操作。而在debug模式下,timer對象的生成期是方法的結束,這樣做也是為了調(diào)試的方便。要不然在調(diào)試時,我們執(zhí)行到Timer timer = new Timer()后想看timer的值時,已經(jīng)被垃圾回收器給回收了,這是我們不期望看到的結果,編譯器如何處理的,我們可以看看編譯器在release模式下和debug模式下對上面的代碼編譯后生成的IL對比我們既知結果。

release模式編譯生成的IL:

.method private hidebysig static void Main(string[] args) cil managed
{
 .entrypoint
 // Code size    32 (0x20)
 .maxstack 8
 IL_0000: ldnull
 IL_0001: ldftn   void GCTest.Program::TimerCallback(object)
 IL_0007: newobj   instance void [mscorlib]System.Threading.TimerCallback::.ctor(object,
                                           native int)
 IL_000c: ldnull
 IL_000d: ldc.i4.0
 IL_000e: ldc.i4   0x7d0
 IL_0013: newobj   instance void [mscorlib]System.Threading.Timer::.ctor(class [mscorlib]System.Threading.TimerCallback,
                                       object,
                                       int32,
                                       int32)
 IL_0018: pop
 IL_0019: call    string [mscorlib]System.Console::ReadLine()
 IL_001e: pop
 IL_001f: ret
} // end of method Program::Main

debug模式下生成的IL:

method private hidebysig static void Main(string[] args) cil managed
{
 .entrypoint
 // Code size    33 (0x21)
 .maxstack 4
 .locals init ([0] class [mscorlib]System.Threading.Timer timer)
 IL_0000: nop
 IL_0001: ldnull
 IL_0002: ldftn   void GCTest.Program::TimerCallback(object)
 IL_0008: newobj   instance void [mscorlib]System.Threading.TimerCallback::.ctor(object,
                                           native int)
 IL_000d: ldnull
 IL_000e: ldc.i4.0
 IL_000f: ldc.i4   0x7d0
 IL_0014: newobj   instance void [mscorlib]System.Threading.Timer::.ctor(class [mscorlib]System.Threading.TimerCallback,
                                       object,
                                       int32,
                                       int32)
 IL_0019: stloc.0
 IL_001a: call    string [mscorlib]System.Console::ReadLine()
 IL_001f: pop
 IL_0020: ret
} // end of method Program::Main

從生成的IL中我們可以看出在debug模式下,生成IL比在release模式下多了19行紅色字體的IL指令碼,該指令碼的作用是將15行生成的引用Timer對象的棧上的變量存放到局部變量0中。所以使得在debug模式下該t還被引用,不能夠回收Timer對象,所以也能出現(xiàn)我們期盼的結果,那么如何在兩種模式下都能得到我們期盼的結果呢。我們可以如下操作。

正確的代碼:

class Program
  {
    static void Main(string[] args)
    {
      Timer timer = new Timer(TimerCallback,null,0,2000);
    
      Console.ReadLine();
      timer.Dispose();
    }

    private static void TimerCallback(object o)
    {
      Console.WriteLine("in TimerCallback method");

      GC.Collect();

      
    }
  }

這時不管是在release模式下還是debug模式下,都會每隔2秒鐘調(diào)用我們的回調(diào)方法。

相關文章

  • 綁定winform中DataGrid

    綁定winform中DataGrid

    綁定winform中DataGrid,需要的朋友可以參考一下
    2013-02-02
  • C#循環(huán)與循環(huán)控制的表達式樹實現(xiàn)

    C#循環(huán)與循環(huán)控制的表達式樹實現(xiàn)

    這篇文章介紹了C#循環(huán)與循環(huán)控制的表達式樹實現(xiàn),文中通過示例代碼介紹的非常詳細。對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2022-01-01
  • c#中利用Tu Share獲取股票交易信息

    c#中利用Tu Share獲取股票交易信息

    這篇文章主要介紹了c#中利用Tu Share獲取股票交易信息,本文通過實例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2020-06-06
  • C#函數(shù)out多個返回值問題

    C#函數(shù)out多個返回值問題

    這篇文章主要介紹了C#函數(shù)out多個返回值問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2023-02-02
  • unity實現(xiàn)方向盤轉動效果

    unity實現(xiàn)方向盤轉動效果

    這篇文章主要為大家詳細介紹了unity實現(xiàn)方向盤轉動效果,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-09-09
  • C#文件上傳與下載的實現(xiàn)方法

    C#文件上傳與下載的實現(xiàn)方法

    這篇文章主要為大家詳細介紹了C#文件上傳與下載的實現(xiàn)方法,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-08-08
  • C# Socket網(wǎng)絡編程實例

    C# Socket網(wǎng)絡編程實例

    這篇文章主要介紹了C# Socket網(wǎng)絡編程實例,分析了Socket網(wǎng)絡通信的原理與具體應用技巧,具有一定參考借鑒價值,需要的朋友可以參考下
    2015-01-01
  • C#泛型詳解及關鍵字作用

    C#泛型詳解及關鍵字作用

    這篇文章主要來講講c#中的泛型,因為泛型在c#中有很重要的位置,對于寫出高可讀性,高性能的代碼有著關鍵的作用,大家都知道泛型公共語言運行庫是非常重要功能,那么為什么使用泛型呢,帶著這個問題一起通過本文學習下吧
    2021-08-08
  • C#繪圖基本方法實例總結

    C#繪圖基本方法實例總結

    C#要實現(xiàn)簡單的畫圖功能可以利用Graphics這個類,下面這篇文章主要給大家介紹了關于C#繪圖基本方法的相關資料,文中通過示例代碼介紹的非常詳細,需要的朋友可以參考下
    2022-12-12
  • C#向無窗口的進程發(fā)送消息

    C#向無窗口的進程發(fā)送消息

    這篇文章主要介紹了C#向無窗口的進程發(fā)送消息 的相關資料,需要的朋友可以參考下
    2016-05-05

最新評論