C#中的Task.Delay()和Thread.Sleep()區別(代碼案例)
一、簡介
1.Thread.Sleep()是同步延遲,Task.Delay()是異步延遲。
2.Thread.Sleep()會阻塞線程,Task.Delay()不會。
3.Thread.Sleep()不能取消,Task.Delay()可以。
4.Task.Delay()實質創建一個運行給定時間的任務,Thread.Sleep()使當前線程休眠給定時間。
5.反編譯Task.Delay(),基本上講它就是個包裹在任務中的定時器。
6.Task.Delay()和Thread.Sleep()最大的區別是Task.Delay()旨在異步運行,在同步代碼中使用Task.Delay()是沒有意義的;在異步代碼中使用Thread.Sleep()是一個非常糟糕的主意。通常使用await關鍵字調用Task.Delay()。
二、代碼案例
案例一:Thread.Sleep()和Task.Delay()比較
代碼:
static void Main(string[] args) { //阻塞,出現CPU等待... Task.Factory.StartNew(delegate { Console.WriteLine(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff") + " ****** Start Sleep()******"); for (int i = 1; i <=10; i++) { Console.WriteLine(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff") + "******Sleep******==>" + i); Thread.Sleep(1000);//同步延遲,阻塞一秒 } Console.WriteLine(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff") + " ******End Sleep()******"); Console.WriteLine(); }); //不阻塞 Task.Factory.StartNew(() => { Console.WriteLine(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff") + " ======StartDelay()======"); for (int i =1; i <=10; i++) { Console.WriteLine(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff") + " ======Delay====== ==>" + i); Task.Delay(1000);//異步延遲 } Console.WriteLine(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff") + " ======End Delay()======"); Console.WriteLine(); }); Console.ReadLine(); Console.ReadKey(); }
結果:
通過運行結果截圖對比看出,Thread.Sleep()是同步延遲,Task.Delay()是異步延遲。
案例二:通過async/await實現Task.Delay()同步
代碼:
//該段代碼通過async/awatit實現“同步”Delay static void Main(string[] args) { Task.Factory.StartNew(async () => { Console.WriteLine(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff") + " ======Start Delay()======"); for (int i = 1; i <=10; i++) { Console.WriteLine(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff") + " ======Delay======" + i); await Task.Delay(1000); } Console.WriteLine(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff") + " ======End Delay()======"); }); Console.ReadKey(); }
結果:
運行結果可以看出,通過async/await實現瞭Task.Delay()同步
案例三:Task.Delay()取消
代碼:
class Program { #region CancellationTokenSource cts = new CancellationTokenSource(); void PutThreadSleep() { Thread.Sleep(5000); } async Task PutTaskDelay() { try { await Task.Delay(5000, cts.Token);//需要.net4.5的支持 } catch (TaskCanceledException ex) { Console.WriteLine(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff ==>") + ex.ToString()); } } private void ThreadSleep() { PutThreadSleep(); Console.WriteLine("Sleep : I am back"); } private async void TaskDelay() { await PutTaskDelay(); Console.WriteLine("Delay : I am back"); } private void CancelTaskDelay() { cts.Cancel(); } #endregion static void Main(string[] args) { #region Program p = new Program(); //不可取消 p.ThreadSleep(); //可取消 p.TaskDelay(); p.CancelTaskDelay(); #endregion Console.ReadKey(); } }
結果:
Task.Delay()取消,拋出異常信息。
三、總結
Task.Delay(),async/await和CancellationTokenSource組合起來使用可以實現可控制的異步延遲。
以上就是C#中的Task.Delay()和Thread.Sleep()區別(代碼案例)的詳細內容,更多關於C# Task.Delay()和Thread.Sleep()的資料請關註WalkonNet其它相關文章!
推薦閱讀:
- c# Task.Wait()與awaiat Task異常處理的區別說明
- C#中async和await的深入分析
- .Net中Task Parallel Library的基本用法
- C#代碼延時的幾種實現
- C# 並行和多線程編程——認識和使用Task