




下載本文檔
版權(quán)說(shuō)明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請(qǐng)進(jìn)行舉報(bào)或認(rèn)領(lǐng)
文檔簡(jiǎn)介
第C#異步編程由淺入深(三)之詳解Awaiter上一篇末尾提到了Awaiter這個(gè)類型,上一篇說(shuō)了,能await的對(duì)象,必須包含GetAwaiter()方法,不清楚的朋友可以看上篇文章。那么,Awaiter到底有什么特別之處呢?
首先,從上篇文章我們知道,一個(gè)Awaiter必須實(shí)現(xiàn)INotifyCompletion接口,這個(gè)接口定義如下:
namespaceSystem.Runtime.CompilerServices
///summary
///Representsanoperationthatwillschedulecontinuationswhentheoperationcompletes.
////summary
publicinterfaceINotifyCompletion
///summarySchedulesthecontinuationactiontobeinvokedwhentheinstancecompletes./summary
///paramname="continuation"Theactiontoinvokewhentheoperationcompletes./param
///exceptioncref="System.ArgumentNullException"Theparamrefname="continuation"/argumentisnull(NothinginVisualBasic)./exception
voidOnCompleted(Actioncontinuation);
}
除此之外還必須包含IsCompleted屬性和包含GetResult()方法。
注意OnCompleted的參數(shù)是一個(gè)Action委托,并且不出意外的話,委托里面總會(huì)有一個(gè)地方調(diào)用一個(gè)MoveNext()方法,它推動(dòng)狀態(tài)機(jī)到達(dá)下一個(gè)狀態(tài),然后執(zhí)行下一個(gè)狀態(tài)需要執(zhí)行的代碼。
那么,知道這個(gè)有什么用呢?第一,它是你充分了解async/await這套機(jī)制的基礎(chǔ),包括與之相關(guān)的同步上下文、執(zhí)行上下文、死鎖問(wèn)題等,第二,它可以實(shí)現(xiàn)一些特殊的功能。
從上一篇我們知道,OnCompleted中的contination的主要目的是推動(dòng)狀態(tài)機(jī)的執(zhí)行,也就是推動(dòng)異步方法中await后面部分的代碼執(zhí)行。從這里看出,continuation的執(zhí)行是受我們控制的,因此我們可以直接執(zhí)行它,或是等待某個(gè)條件成熟然后執(zhí)行它,我們可以把它放到線程池執(zhí)行,也可以單獨(dú)起一個(gè)線程執(zhí)行。譬如,我們可以讓await后面部分的代碼直接在線程池上執(zhí)行。
publicstaticasyncTaskAwaiterTest()
Console.WriteLine($"是否是線程池線程?{Thread.CurrentThread.IsThreadPoolThread}");
awaitdefault(SkipToThreadPoolAwaiter);
Console.WriteLine($"是否是線程池線程?{Thread.CurrentThread.IsThreadPoolThread}");
staticvoidMain(string[]args)
_=AwaiterTest();
Console.ReadLine();
publicstructSkipToThreadPoolAwaiter:INotifyCompletion
publicboolIsCompleted=false;
publicvoidGetResult()
Console.WriteLine("調(diào)用GetResult以獲取結(jié)果");
publicvoidOnCompleted(Actioncontinuation)
Console.WriteLine("調(diào)用OnCompleted,把Await后面部分要執(zhí)行的代碼傳遞過(guò)來(lái)(傳遞MoveNext,以推動(dòng)狀態(tài)機(jī)流轉(zhuǎn))");
ThreadPool.QueueUserWorkItem(state=
Console.WriteLine("開(kāi)始執(zhí)行Await后面部分的代碼");
continuation();
Console.WriteLine("后面部分的代碼執(zhí)行完畢");
Console.WriteLine("返回調(diào)用線程");
publicSkipToThreadPoolAwaiterGetAwaiter()
Console.WriteLine("獲得Awaiter");
returnthis;
}
這是一個(gè)控制臺(tái)程序,輸出結(jié)果如下。
是否是線程池線程?False
獲得Awaiter
調(diào)用OnCompleted,把Await后面部分要執(zhí)行的代碼傳遞過(guò)來(lái)(傳遞MoveNext,以推動(dòng)狀態(tài)機(jī)流轉(zhuǎn))
返回調(diào)用線程
開(kāi)始執(zhí)行Await后面部分的代碼
調(diào)用GetResult以獲取結(jié)果
是否是線程池線程?True
后面部分的代碼執(zhí)行完畢
特別注意一下,第五步說(shuō)明可能有點(diǎn)疑惑,怎么第六步不是打印是否是線程池線程?原因是部分awaiter是有返回值的,在執(zhí)行await后面部分的代碼時(shí),會(huì)首先調(diào)用GetResult()以獲取結(jié)果。這對(duì)編譯器改造異步方法來(lái)說(shuō)是一個(gè)固定的模式(上篇文章沒(méi)有體現(xiàn)這一步)。
把Awaiter改成有返回值嘗試。
publicstaticasyncTaskAwaiterTest()
Console.WriteLine($"是否是線程池線程?{Thread.CurrentThread.IsThreadPoolThread}");
varres=awaitdefault(SkipToThreadPoolAwaiter);
Console.WriteLine($"結(jié)果是{res}");
staticvoidMain(string[]args)
_=AwaiterTest();
Console.ReadLine();
publicstructSkipToThreadPoolAwaiter:INotifyCompletion
publicboolIsCompleted=false;
publicintGetResult()
Console.WriteLine("調(diào)用GetResult以獲取結(jié)果");
return1;
publicvoidOnCompleted(Actioncontinuation)
Console.WriteLine("調(diào)用OnCompleted,把Await后面部分要執(zhí)行的代碼傳遞過(guò)來(lái)(傳遞MoveNext,以推動(dòng)狀態(tài)機(jī)流轉(zhuǎn))");
ThreadPool.QueueUserWorkItem(state=
Console.WriteLine("開(kāi)始執(zhí)行Await后面部分的代碼");
continuation();
Console.WriteLine("后面部分的代碼執(zhí)行完畢");
Console.WriteLine("返回調(diào)用線程");
publicSkipToThreadPoolAwaiterGetAwaiter()
Console.WriteLine("獲得Awaiter");
returnthis;
輸出如下
是否是線程池線程?False
獲得Awaiter
調(diào)用OnCompleted,把Await后面部分要執(zhí)行的代碼傳遞過(guò)來(lái)(傳遞MoveNext,以推動(dòng)狀態(tài)機(jī)流轉(zhuǎn))
返回調(diào)用線程
開(kāi)始執(zhí)行Await后面部分的代碼
調(diào)用GetResult以獲取結(jié)果
結(jié)果是1
是否是線程池線程?True
后面部分的代碼執(zhí)行完畢
對(duì)照前面的文章來(lái)看,相信你應(yīng)該有所得,能解決你部分的疑惑。前面說(shuō)到,我們可以控制continuation的執(zhí)行,那如果當(dāng)前線程有同步上下文(SychronizationContext),我們是不是可以放到同步上下文中執(zhí)行?TaskAwaiter是會(huì)這么做的,如果你不想它使用同步上下文,你可以在Task實(shí)例上調(diào)用ConfigureAwait(false),它表面后面部分的代碼將不會(huì)使用同步上下文執(zhí)行。
另外說(shuō)一下Task.Yield()這個(gè)Awaiter,他的行為是捕捉同步上下文,如果有,則會(huì)放到同步上下文中執(zhí)行,如果沒(méi)有,則會(huì)放到線程池中執(zhí)行。在窗體程序中,有時(shí)候你打開(kāi)一個(gè)模態(tài)對(duì)話框,會(huì)導(dǎo)致主窗體部分的動(dòng)畫(huà)沒(méi)有反應(yīng),在模態(tài)對(duì)話框關(guān)閉之后,才會(huì)反應(yīng)。原因是模態(tài)對(duì)話框阻塞了主窗體的消息
溫馨提示
- 1. 本站所有資源如無(wú)特殊說(shuō)明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請(qǐng)下載最新的WinRAR軟件解壓。
- 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請(qǐng)聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
- 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁(yè)內(nèi)容里面會(huì)有圖紙預(yù)覽,若沒(méi)有圖紙預(yù)覽就沒(méi)有圖紙。
- 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
- 5. 人人文庫(kù)網(wǎng)僅提供信息存儲(chǔ)空間,僅對(duì)用戶上傳內(nèi)容的表現(xiàn)方式做保護(hù)處理,對(duì)用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對(duì)任何下載內(nèi)容負(fù)責(zé)。
- 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請(qǐng)與我們聯(lián)系,我們立即糾正。
- 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時(shí)也不承擔(dān)用戶因使用這些下載資源對(duì)自己和他人造成任何形式的傷害或損失。
最新文檔
- 醫(yī)療器械行業(yè)中的品牌建設(shè)與市場(chǎng)定位
- 醫(yī)療器械技術(shù)進(jìn)步對(duì)商業(yè)發(fā)展的影響
- 醫(yī)療行業(yè)中的科技力量-探討移支動(dòng)技術(shù)在健康教育中的應(yīng)用
- 利用區(qū)塊鏈進(jìn)行精確控制優(yōu)化醫(yī)療資源供應(yīng)鏈管理的路徑研究
- 2025年生產(chǎn)部工作年度總結(jié)模版
- 醫(yī)療健康產(chǎn)業(yè)中的大數(shù)據(jù)與人工智能
- 醫(yī)療大數(shù)據(jù)分析在疾病預(yù)防中的價(jià)值
- 企業(yè)食堂供貨合同范例
- 以環(huán)保為目標(biāo)創(chuàng)新型醫(yī)院的設(shè)計(jì)與實(shí)施
- 供銷合同范例6
- 光刻光源創(chuàng)新-全面剖析
- 人工智能在環(huán)保領(lǐng)域的應(yīng)用及挑戰(zhàn)
- 2025年陜西省初中學(xué)業(yè)水平考試英語(yǔ) 例析與指導(dǎo) 試卷示例題答案及聽(tīng)力材料
- 泉州地理會(huì)考題目及答案
- 2025年工會(huì)知識(shí)競(jìng)賽題庫(kù)200題及答案(完整版)
- 小學(xué)教育學(xué)(第5版)課件 第6章 小學(xué)教育內(nèi)容;第7章 小學(xué)教育活動(dòng)
- 完整版高中古詩(shī)文必背72篇【原文+注音+翻譯】
- 主題班會(huì)課件-《花開(kāi)應(yīng)有時(shí)》預(yù)防早戀男女交往
- 2025年安全生產(chǎn)考試題庫(kù):船舶修造企業(yè)安全規(guī)范試題
- 2025年武漢數(shù)學(xué)四調(diào)試題及答案
- T-CI 622-2024 家庭母乳采集儲(chǔ)運(yùn)要求
評(píng)論
0/150
提交評(píng)論