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組合起來使用可以實作可控制的異步延遲,
轉載請註明出處,本文鏈接:https://www.uj5u.com/net/285377.html
標籤:.NET技术
上一篇:Net5學習筆記
