Task 非同步小技巧

一事冇誠發表於2021-10-15

原文地址:Task 非同步小技巧 - 一事冇誠 - 部落格園 (cnblogs.com)

async Task 語法糖出來後,非同步程式設計變得非常簡單,適合需要耗費較長時間的任務。

有些小夥伴使用後可能會非常疑惑,使用非同步和同步,在耗時上幾乎沒有差別。

下面我們看一個例子,場景是需要呼叫多個第三方的WebApi,分別是獲取名稱、年齡、性別,由於網路環境等原因,api響應時間可能會接近1秒

 1 public async Task Test()
 2 {
 3     var sw = new Stopwatch();
 4     sw.Start();
 5 
 6     var userName = await GetUserNameAsync();
 7     var userAge = await GetUserAgeAsync();
 8     var userSex = await GetUserSexAsync();
 9 
10     sw.Stop();
11     var ts = sw.Elapsed;
12     Console.WriteLine($"總共耗時:{ts.TotalMilliseconds}ms");
13 }
14 
15 private async Task<string> GetUserNameAsync()
16 {
17     await Task.Delay(500);
18     return "小明";
19 }
20 
21 private async Task<string> GetUserAgeAsync()
22 {
23     await Task.Delay(800);
24     return "11";
25 }
26 
27 private async Task<string> GetUserSexAsync()
28 {
29     await Task.Delay(900);
30     return "11";
31 }

執行後發現,這個時間2秒多,這使用者體驗肯定是無法忍受的

 導致這樣結果的原因是每次進行非同步呼叫的時候,都在非同步函式前加上了 await ,這會導致該執行緒阻塞,等待直到結果返回,每個非同步函式都await,時間自然就疊加了,為了解決這個問題,使用一個小技巧,可以將程式碼改成下面這樣

 1 public async Task Test()
 2 {
 3     var sw = new Stopwatch();
 4     sw.Start();
 5 
 6     var userNameTask =  GetUserNameAsync();
 7     var userAgeTask =  GetUserAgeAsync();
 8     var userSexTask =  GetUserSexAsync();
 9 
10     var userName = await userNameTask;
11     var userAge = await userAgeTask;
12     var userSex = await userSexTask;
13 
14     sw.Stop();
15     var ts = sw.Elapsed;
16     Console.WriteLine($"總共耗時:{ts.TotalMilliseconds}ms");
17 }
18 
19 private async Task<string> GetUserNameAsync()
20 {
21     await Task.Delay(500);
22     return "小明";
23 }
24 
25 private async Task<string> GetUserAgeAsync()
26 {
27     await Task.Delay(800);
28     return "11";
29 }
30 
31 private async Task<string> GetUserSexAsync()
32 {
33     await Task.Delay(900);
34     return "11";
35 }

這次執行的總耗時,就是3個非同步中,耗時最長那個 GetUserSexAsync

 為什麼會這樣呢,這個小技巧的關鍵是這裡,當執行到非同步函式的時候,不加 await,不進行等待,這樣就不會造成阻塞,讓這些任務乖乖在別的執行緒的執行,當需要用到他們的時候,再去等待返回值,所以時間上不會進行疊加,哪個最長,總耗時就是哪個

1 var userNameTask =  GetUserNameAsync();
2 var userAgeTask =  GetUserAgeAsync();
3 var userSexTask =  GetUserSexAsync();
4 
5 var userName = await userNameTask;
6 var userAge = await userAgeTask;
7 var userSex = await userSexTask;

 

相關文章