C#中Task.Yield的用途深入講解
前言
最近在閱讀 .NET Threadpool starvation, and how queuing makes it worse 這篇博文時(shí)發(fā)現(xiàn)文中代碼中的一種 Task 用法之前從未見(jiàn)過(guò),在網(wǎng)上看了一些資料后也是云里霧里不知其解,很是困擾。今天在程序員節(jié)的大好日子里終于想通了,于是寫(xiě)下這篇隨筆分享給大家,也過(guò)過(guò)專心寫(xiě)博客的癮。
這種從未見(jiàn)過(guò)的用法就是下面代碼中的 await Task.Yield() :
static async Task Process()
{
await Task.Yield();
var tcs = new TaskCompletionSource<bool>();
Task.Run(() =>
{
Thread.Sleep(1000);
tcs.SetResult(true);
});
tcs.Task.Wait();
}
(注:上面的代碼不是示例,只是因?yàn)檫@段代碼而初遇 await Task.Yield)
Task.Yield 簡(jiǎn)單來(lái)說(shuō)就是創(chuàng)建時(shí)就已經(jīng)完成的 Task ,或者說(shuō)執(zhí)行時(shí)間為0的 Task ,或者說(shuō)是空任務(wù),也就是在創(chuàng)建時(shí)就將 Task 的 IsCompeted 值設(shè)置為0。
那 await 一個(gè)空任務(wù)會(huì)怎樣?我們知道在 await 時(shí)會(huì)釋放當(dāng)前線程,等所 await 的 Task 完成時(shí)會(huì)從線程池中申請(qǐng)新的線程繼續(xù)執(zhí)行 await 之后的代碼,這本來(lái)是為了解決異步操作(比如IO操作)霸占線程實(shí)際卻用不到線程的問(wèn)題,而 Task.Yield 卻產(chǎn)生了一個(gè)不僅沒(méi)有異步操作而且什么也不干的 Task ,不是吃飽了撐著嗎?
今天吃晚飯的時(shí)候終于想明白了——吃飽了沒(méi)有撐。Task.Yield 產(chǎn)生的空任務(wù)僅僅是為 await 做嫁衣,而真正的圖謀是借助 await 實(shí)現(xiàn)線程的切換,讓 await 之后的操作重新排隊(duì)從線程池中申請(qǐng)線程繼續(xù)執(zhí)行。
這樣做有什么好處呢?
線程是非常非常寶貴的資源,千金難買一線程,而且有優(yōu)先級(jí),提高線程利用率的重要手段之一就是及時(shí)將線程分配給最需要的地方,而最奢侈的之一是讓一個(gè)優(yōu)先級(jí)低執(zhí)行時(shí)間長(zhǎng)的操作一直占用著一個(gè)線程,await Task.Yield 可以讓你巧妙地借助 await 的線程切換能力,將不太重要的比較耗時(shí)的操作放在新的線程(重新排隊(duì)從線程池中申請(qǐng)到的線程)中執(zhí)行。打個(gè)比方,很多人排隊(duì)在外婆家就餐,你來(lái)的時(shí)候比較巧,正好有位置,但你本來(lái)就不著急肚子也不太餓準(zhǔn)備慢慢吃慢慢聊,而排隊(duì)的人當(dāng)中有些人很餓很著急吃完還有事,這時(shí)你如果先點(diǎn)幾個(gè)招牌菜解解饞,然后將座位讓出來(lái),重新排隊(duì),并且排隊(duì)的人當(dāng)中像你這樣的都這么做,那些排隊(duì)中心急如焚的人真是是幸福感爆棚,外婆家的老板也笑彎了腰。你讓出座位重新排隊(duì)的愛(ài)心行為就是 await Task.Yield() 。
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)腳本之家的支持。
相關(guān)文章
C#導(dǎo)出數(shù)據(jù)到Excel文件的方法
這篇文章主要介紹了C#導(dǎo)出數(shù)據(jù)到Excel文件的方法,涉及C#操作Excel的相關(guān)技巧,非常具有實(shí)用價(jià)值,需要的朋友可以參考下2015-04-04
C#正則表達(dá)式獲取下拉菜單(select)的相關(guān)屬性值
這篇文章主要介紹了C#正則表達(dá)式獲取下拉菜單(select)的相關(guān)屬性值,比如可以獲得name屬性的值、value值、指定值,需要的朋友可以參考下2014-07-07
使用代理模式來(lái)進(jìn)行C#設(shè)計(jì)模式開(kāi)發(fā)的基礎(chǔ)教程
這篇文章主要介紹了使用代理模式來(lái)進(jìn)行C#設(shè)計(jì)模式開(kāi)發(fā)的基礎(chǔ)教程,代理模式主張?jiān)诳蛻舳撕湍繕?biāo)對(duì)象中間建立中介來(lái)降低程序設(shè)計(jì)的耦合度,需要的朋友可以參考下2016-02-02
使用C#表達(dá)式樹(shù)實(shí)現(xiàn)對(duì)象的深克隆(實(shí)例詳解)
C# 的表達(dá)式樹(shù)提供了一個(gè)強(qiáng)大的機(jī)制,可以將代碼以數(shù)據(jù)結(jié)構(gòu)的形式表示出來(lái),使得代碼可以在運(yùn)行時(shí)進(jìn)行檢查、修改或執(zhí)行,這為動(dòng)態(tài)查詢生成、代碼優(yōu)化和動(dòng)態(tài)編程提供了很多可能性,這篇文章主要介紹了使用C#強(qiáng)大的表達(dá)式樹(shù)實(shí)現(xiàn)對(duì)象的深克隆,需要的朋友可以參考下2024-05-05
C#實(shí)現(xiàn)線程安全的簡(jiǎn)易日志記錄方法
這篇文章主要介紹了C#實(shí)現(xiàn)線程安全的簡(jiǎn)易日志記錄方法,比較實(shí)用的功能,需要的朋友可以參考下2014-08-08

