c# Task.Wait()與awaiat Task異常處理的區別說明
Task.Wait()與awaiat Task異常處理區別
Task異常處理
下面有兩個例子代碼,可以直接復制粘貼到.net core中運行。兩個代碼要實現的功能完全一樣,但是內核卻又很大差異。
先看下面用await的例子與輸出:
using System; using System.Threading; using System.Threading.Tasks; class Program { static async Task Main() { System.Console.WriteLine($"Main Task ID:{Thread.CurrentThread.ManagedThreadId}"); var task = Task.Run(() => { System.Console.WriteLine($"In Task.Run(), Task ID:{Thread.CurrentThread.ManagedThreadId}"); int[] vary=new int[5]; while (true) { Thread.Sleep(3000); int d = vary[6]; } }); // Just continue on this thread, or await with try-catch: try { await task; } catch (IndexOutOfRangeException ex) { System.Console.WriteLine(ex.Message); System.Console.WriteLine($"After Wait(), Task ID:{Thread.CurrentThread.ManagedThreadId}"); } catch(AggregateException ex) { System.Console.WriteLine(ex.Message); System.Console.WriteLine($"After Wait(), Task ID:{Thread.CurrentThread.ManagedThreadId}"); } finally { //... } System.Console.WriteLine("Reach end."); Console.ReadKey(); } } /* Main Task ID:1 In Task.Run(), Task ID:4 Index was outside the bounds of the array. Catch System.IndexOutOfRangeException After Wait(), Task ID:4 Reach end. */
再看Task.Wait()方法下的異常處理與輸出:
using System; using System.Threading; using System.Threading.Tasks; class Program { static void Main() { System.Console.WriteLine($"Main Task ID:{Thread.CurrentThread.ManagedThreadId}"); var task = Task.Run(() => { System.Console.WriteLine($"In Task.Run(), Task ID:{Thread.CurrentThread.ManagedThreadId}"); int[] vary=new int[5]; while (true) { Thread.Sleep(3000); int d = vary[6]; } }); // Just continue on this thread, or await with try-catch: try { task.Wait(); } catch (IndexOutOfRangeException ex) { System.Console.WriteLine($"Catch {ex.GetType()}"); System.Console.WriteLine($"After Wait(), Task ID:{Thread.CurrentThread.ManagedThreadId}"); } catch(AggregateException ex) { System.Console.WriteLine($"Catch {ex.GetType()}"); System.Console.WriteLine($"After Wait(), Task ID:{Thread.CurrentThread.ManagedThreadId}"); } finally { //... } System.Console.WriteLine("Reach end."); Console.ReadKey(); } } /* Main Task ID:1 In Task.Run(), Task ID:4 Catch System.AggregateException One or more errors occurred. (Index was outside the bounds of the array.) After Wait(), Task ID:1 Reach end. */
從例子中可以看出,await之後的代碼其實都是在新的線程(4線程)中執行,而Task.Wait()方法後的線程則是在主線程中執行。
因此,await之後的代碼完全以傳統方式處理異常;而Task.Wait()拋出的異常則由於是從新線程往外部線程拋出,所以它是被重新封裝為AggregateException異常拋出。
Task.WaitAll()註意事項
使用Task.WaitAll() 等待多任務執行完畢的時候發現,等待的任務還沒結束,Task.WaitAll() 就先結束瞭,於是就寫瞭一段測試代碼進行驗證。
先上代碼
static void Main(string[] args) { //建立兩個任務 Task t1 = new Task(async () => await T1()); Task t2 = new Task(async () => await T2()); //啟動任務 t1.Start(); t2.Start(); //等待任務完成 Task.WaitAll(t1, t2); Print("WaitAll Done"); Console.ReadLine(); } static async Task T1() { Print("T1 Start"); Thread.Sleep(1000); Print("T1 await"); await Task.Delay(1000); Print("T1 Done"); } static async Task T2() { Print("T2 Start"); Thread.Sleep(1000); Print("T2 await"); await Task.Delay(1000); Print("T2 Done"); } static void Print(string msg) { Console.WriteLine($"{DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.ffffff")}: {msg}"); }
再上結果,註意看T1、T2 Done 和 WaitAll Done的打印時間:
果然,坑!
Task.WaitAll() 盡然比等待的Task先結束。
總結:(不推薦,請看補充內容)
new Task().Start() 中一旦使用 await ,會立馬返回結束狀態。
所以,在使用 Task.WaitAll() 或其接續任務的時候,可以考慮使用 Thead.sleep() 替代 await Task.Delay() 。
2022-04-25 補充:
經過【32號就放假】提醒,測試瞭Task.Run() 和 Task.Factory.StartNew()兩個方法,得出結論:
1、 在Task.Run()啟動任務中,await會正常運行;(推薦使用)
static void Main(string[] args) { //建立兩個任務 Task t1 = Task.Run(T1); Task t2 = Task.Run(T2); //等待任務完成 Task.WaitAll(t1, t2); Print("WaitAll Done"); Console.ReadLine(); }
2、在Task.Factory.StartNew() 啟動任務中,會立馬返回結束狀態。
static void Main(string[] args) { //建立兩個任務 Task t1 = Task.Factory.StartNew(T1); Task t2 = Task.Factory.StartNew(T2); //等待任務完成 Task.WaitAll(t1, t2); Print("WaitAll Done"); Console.ReadLine(); }
以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。
推薦閱讀:
- .NET實現異步編程async和await
- C#中async和await的深入分析
- .Net中Task Parallel Library的基本用法
- C# 並行和多線程編程——認識和使用Task
- C# 並行和多線程編程——Task進階知識