關(guān)于async和await的一些誤區(qū)實(shí)例詳解
微軟官方的MSDN上說(shuō)async和await是“異步”,但是不少人(包括筆者自己)都有一些誤區(qū)需要澄清:為什么await語(yǔ)句之后沒(méi)有執(zhí)行?不是異步嗎?
先舉一個(gè)示例代碼如下:
public partial class Form1 : Form { public async Task Processing() { await Task.Delay(5000); label1.Text = "Succuessful"; } public Form1() { InitializeComponent(); } private async void button1_Click(object sender, EventArgs e) { await Processing(); MessageBox.Show("Button's event completed"); } }
很多人(包括筆者)一開(kāi)始會(huì)覺(jué)得異步好像類似多線程一樣,到await的時(shí)候會(huì)在后臺(tái)先開(kāi)啟一個(gè)線程執(zhí)行任務(wù),隨后主線程(這里是UI線程)將自動(dòng)執(zhí)行后面的部分(即彈出“Button's event completed”的消息框)。
其實(shí)這個(gè)理解是錯(cuò)誤的。async和await的本質(zhì)其實(shí)是“yield return”和“LINQ”的“迭代式”等待。我們應(yīng)該清楚一點(diǎn):那就是你寫了LINQ語(yǔ)句:
var results = from …… select ……; foreach(var r in results) { …… }
當(dāng)你下斷點(diǎn)你會(huì)發(fā)覺(jué)results并不會(huì)立即執(zhí)行,直到使用到results的地方(例子中也就是foreach這里)才會(huì)被執(zhí)行(此時(shí)黃色跟蹤調(diào)試的光棒又會(huì)折回到var results……這里,然后等到results執(zhí)行完畢之后才真正進(jìn)入foreach進(jìn)行執(zhí)行)。
所以,async/await和LINQ的這種“迭代式”的“異步操作”是異曲同工的。只不過(guò)async/await本質(zhì)是返回一個(gè)Task而已,而Task又是異步的(因?yàn)門ask本質(zhì)就是一個(gè)線程),所以真正執(zhí)行到(使用到async方法的時(shí)候)帶有await的方法的時(shí)候,后臺(tái)才會(huì)真正開(kāi)啟一個(gè)線程去執(zhí)行任務(wù)。此時(shí)主線程會(huì)等待這個(gè)Task線程直到其執(zhí)行完畢(IsComplete屬性為True為止)。所以界面是不會(huì)卡頓的。
所以,await是Task的異步等待而已,并不是我們所謂的“異步操作”;拿它和LINQ作對(duì)比,你會(huì)發(fā)現(xiàn)LINQ執(zhí)行順序和它一致,只不過(guò)LINQ沒(méi)有異步等待(當(dāng)然沒(méi)有!又沒(méi)有開(kāi)啟線程啥的……)。
我們進(jìn)一步可以這樣對(duì)比:
LINQ:變量 = LINQ語(yǔ)句(表達(dá)式)
等到使用LINQ變量的時(shí)候才折返到LINQ語(yǔ)句處真正執(zhí)行LINQ語(yǔ)句。
異步等待:變量 = 異步方法
等到使用await+異步方法的時(shí)候才會(huì)折返到該異步方法處,開(kāi)啟線程真正執(zhí)行異步方法,主線程被掛起(但不會(huì)造成界面死掉),直至子線程Task任務(wù)完全執(zhí)行完畢為止。
在LINQ中,你如果需要立即執(zhí)行,可以使用擴(kuò)展方法:
var results = (from ……
select ……).ToList();
因?yàn)榱⒓词褂玫搅诉@個(gè)LINQ語(yǔ)句,所以會(huì)被立即執(zhí)行。
同樣地,異步等待也可以變成類似Wait一樣的同步等待:
private async void button1_Click(object sender, EventArgs e) { Processing().GetAwaiter().GetResult(); MessageBox.Show("Button's event completed"); }
因?yàn)镻rocessing本來(lái)就返回Task,當(dāng)然也可以使用Wait進(jìn)行同步等待。
相關(guān)文章
C#使用Clipboard類實(shí)現(xiàn)剪貼板功能
這篇文章介紹了C#使用Clipboard類實(shí)現(xiàn)剪貼板功能的方法,文中通過(guò)示例代碼介紹的非常詳細(xì)。對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-06-06C#中Dictionary泛型集合7種常見(jiàn)的用法
本文主要介紹了Dictionary集合的7種最基礎(chǔ)的用法,包括創(chuàng)建、添加、查找、遍歷、刪除等方法,程序都是由簡(jiǎn)入繁,希望能通過(guò)閱讀簡(jiǎn)單的示例,給大家一些啟發(fā)。2016-03-03C#實(shí)現(xiàn)Winform無(wú)邊框移動(dòng)的方法
這篇文章主要介紹了C#實(shí)現(xiàn)Winform無(wú)邊框移動(dòng)的方法,涉及C#針對(duì)WinForm窗口操作的相關(guān)技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-09-09C#實(shí)現(xiàn)簡(jiǎn)單的loading提示控件實(shí)例代碼
本文通過(guò)實(shí)例代碼給大家介紹了C#實(shí)現(xiàn)簡(jiǎn)單的loading提示控件功能,代碼非常簡(jiǎn)單,具有參考借鑒價(jià)值,需要的朋友參考下吧2017-09-09C#實(shí)現(xiàn)簡(jiǎn)易的加密、解密字符串工具類實(shí)例
這篇文章主要介紹了C#實(shí)現(xiàn)簡(jiǎn)易的加密、解密字符串工具類,涉及C#字符串加密與加密的實(shí)現(xiàn)技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-08-08