Task提高異步執(zhí)行效率技巧
async Task 語法糖出來后,異步編程變得非常簡單,適合需要耗費較長時間的任務。
有些小伙伴使用后可能會非常疑惑,使用異步和同步,在耗時上幾乎沒有差別。
下面我們看一個例子,場景是需要調用多個第三方的WebApi,分別是獲取名稱、年齡、性別,由于網絡環(huán)境等原因,api響應時間可能會接近1秒
public async Task Test() { var sw = new Stopwatch(); sw.Start(); var userName = await GetUserNameAsync(); var userAge = await GetUserAgeAsync(); var userSex = await GetUserSexAsync(); sw.Stop(); var ts = sw.Elapsed; Console.WriteLine($"總共耗時:{ts.TotalMilliseconds}ms"); } private async Task<string> GetUserNameAsync() { await Task.Delay(500); return "小明"; } private async Task<string> GetUserAgeAsync() { await Task.Delay(800); return "11"; } private async Task<string> GetUserSexAsync() { await Task.Delay(900); return "11"; }
運行后發(fā)現,這個時間2秒多,這用戶體驗肯定是無法忍受的
導致這樣結果的原因是每次進行異步調用的時候,都在異步函數前加上了 await ,對于單單這個過程來說,其實相當于同步,等待直到結果返回,每個異步函數都await,時間自然就疊加了,為了解決這個問題,使用一個小技巧,可以將代碼改成下面這樣
public async Task Test() { var sw = new Stopwatch(); sw.Start(); var userNameTask = GetUserNameAsync(); var userAgeTask = GetUserAgeAsync(); var userSexTask = GetUserSexAsync(); var userName = await userNameTask; var userAge = await userAgeTask; var userSex = await userSexTask; sw.Stop(); var ts = sw.Elapsed; Console.WriteLine($"總共耗時:{ts.TotalMilliseconds}ms"); } private async Task<string> GetUserNameAsync() { await Task.Delay(500); return "小明"; } private async Task<string> GetUserAgeAsync() { await Task.Delay(800); return "11"; } private async Task<string> GetUserSexAsync() { await Task.Delay(900); return "11"; }
這次運行的總耗時,就是3個異步中,耗時最長那個GetUserSexAsync
為什么會這樣呢,這個小技巧的關鍵是這里,當執(zhí)行到異步函數的時候,不加 await,不進行等待,讓這些任務乖乖在別的線程的執(zhí)行,當需要用到他們的時候,再去等待返回值,所以時間上不會進行疊加,哪個最長,總耗時就是哪個
var userNameTask = GetUserNameAsync(); var userAgeTask = GetUserAgeAsync(); var userSexTask = GetUserSexAsync(); var userName = await userNameTask; var userAge = await userAgeTask; var userSex = await userSexTask;
到此這篇關于Task提高異步執(zhí)行效率技巧的文章就介紹到這了。希望對大家的學習有所幫助,也希望大家多多支持腳本之家。