版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請進行舉報或認領
文檔簡介
第C#多線程TPL模式高級用法探秘我們先來看下面的一個小示例:一個Winfrom程序,界面上有一個按鈕,有兩個異步方法,點擊按鈕調(diào)用兩個異步方法,彈出執(zhí)行順序,代碼如下:
usingSystem;
usingSystem.Threading;
usingSystem.Threading.Tasks;
usingSystem.Windows.Forms;
namespaceTPLDemoSln
publicpartialclassForm1:Form
publicForm1()
InitializeComponent();
///summary
///按鈕點擊事件
////summary
///paramname="sender"/param
///paramname="e"/param
privateasyncvoidbtnStart_Click(objectsender,EventArgse)
stringi1=awaitF1Async();
MessageBox.Show("i1="+i1);
stringi2=awaitF2Async();
MessageBox.Show("i2="+i2);
///summary
///異步方法F1
////summary
///returns/returns
privateTaskstringF1Async()
MessageBox.Show("F1Start");
returnTask.Runstring(()=
//休眠1秒
Thread.Sleep(1000);
MessageBox.Show("F1Run");
return"F1";
///summary
///異步方法F2
////summary
///returns/returns
privateTaskstringF2Async()
MessageBox.Show("F2Start");
returnTask.Runstring(()=
//休眠2秒
Thread.Sleep(2000);
MessageBox.Show("F2Run");
return"F2";
}
在上面的代碼中,Task.Run()是用來把一個代碼段包裝為TaskT的方法,Run中委托的代碼體就是異步任務執(zhí)行的邏輯,最后return返回值。
運行程序,可以得到如下的輸出順序:
F1Start-F1Run-i1=F1-F2Start-F2Run-i2=F2。
我們對按鈕事件進行修改,修改為下面的代碼,在看看執(zhí)行順序:
///summary
///按鈕點擊事件
////summary
///paramname="sender"/param
///paramname="e"/param
privateasyncvoidbtnStart_Click(objectsender,EventArgse)
//stringi1=awaitF1Async();
//MessageBox.Show("i1="+i1);
//stringi2=awaitF2Async();
//MessageBox.Show("i2="+i2);
Taskstringtask1=F1Async();
Taskstringtask2=F2Async();
stringi1=awaittask1;
MessageBox.Show("i1="+i1);
stringi2=awaittask2;
MessageBox.Show("i2="+i2);
}
再次運行程序,查看輸出順序:
F1Start-F2Start-F1Run-i1=F1-F2Run-i2=F2。
可以看出兩次的執(zhí)行順序不一致。這是什么原因呢?
這是因為并不是到了await才開始執(zhí)行Task異步任務,執(zhí)行到Taskstringtask1=F1Async()這句代碼的時候,F(xiàn)1Async異步任務就開始執(zhí)行了。同理,執(zhí)行到下一句代碼就開始執(zhí)行F2Async異步任務了。await是為了保證執(zhí)行到這里的時候異步任務一定執(zhí)行完。執(zhí)行到await的時候,如果異步任務還沒有執(zhí)行,那么就等待異步任務執(zhí)行完。如果異步任務已經(jīng)執(zhí)行完了,那么就直接獲取異步任務的返回值。
我們上面解釋的原因是否正確呢?我們可以做下面的一個實驗,來驗證上面說的原因,我們把按鈕事件里面的await注釋掉,看看異步方法還會不會執(zhí)行,代碼如下:
///summary
///按鈕點擊事件
////summary
///paramname="sender"/param
///paramname="e"/param
privateasyncvoidbtnStart_Click(objectsender,EventArgse)
//stringi1=awaitF1Async();
//MessageBox.Show("i1="+i1);
//stringi2=awaitF2Async();
//MessageBox.Show("i2="+i2);
Taskstringtask1=F1Async();
Taskstringtask2=F2Async();
//并不是到了await才開始執(zhí)行Task異步任務,這里是保證異步任務一定執(zhí)行完
//代碼執(zhí)行到這里的時候,如果沒有執(zhí)行完就等待執(zhí)行完,如果已經(jīng)執(zhí)行完,就直接獲取返回值
//這里注釋掉await代碼段,查看異步方法是否會執(zhí)行
//stringi1=awaittask1;
//MessageBox.Show("i1="+i1);
//stringi2=awaittask2;
//MessageBox.Show("i2="+i2);
}
運行程序,發(fā)現(xiàn)異步方法還是會執(zhí)行,這就說明我們上面解釋的原因是正確的。感興趣的可以使用Reflector反編譯查看內(nèi)部實現(xiàn)的原理,主要是MoveNext()方法內(nèi)部。
我們可以得到下面的結(jié)論:
只要方法是TaskT類型的返回值,都可以用await來等待調(diào)用獲取返回值。如果一個返回TaskT類型的方法被標記了async,那么只要方法內(nèi)部直接returnT這個類型的實例就可以了。一個返回TaskT類型的方法如果沒有被標記為async,那么需要方法內(nèi)部直接return一個Task的實例。
上面說的第二點看下面的代碼
///summary
///方法標記為async直接返回一個int類型的數(shù)值即可
////summary
///returns/returns
privateasyncTaskintF3Async()
return2;
}
上面的第三點可以看下面的代碼:
///summary
///方法沒有被標記為async,直接返回一個Task
////summary
///returns/returns
privateTaskintF4Async()
returnTask.Runint(()=
return2;
}
二、TPL高級
我們做一些總結(jié):
1、如果方法內(nèi)部有await,則方法必須標記為async。await和async是成對出現(xiàn)的,只有await沒有async程序會報錯。只有async沒有await,程序會按照同步方法執(zhí)行。
2、ASP.NETMVC中的Action方法和WinForm中的事件處理方法都可以標記為async,控制臺的Main()方法不能被標記為async。對于不能標記為async的方法怎么辦呢?我們可以使用Result屬性來獲取值,看下面代碼:
usingSystem;
usingSystem.Net.Http;
usingSystem.Threading.Tasks;
namespaceAsyncDemo
classProgram
staticvoidMain(string[]args)
//實例化對象
HttpClientclient=newHttpClient();
//調(diào)用異步的Get方法
TaskHttpResponseMessagetaskMsg=client.GetAsync("");
//通過Result屬性獲取返回值
HttpResponseMessagemsg=taskMsg.Result;
TaskstringtaskRead=msg.Content.ReadAsStringAsync();
stringhtml=taskRead.Result;
Console.WriteLine(html);
Console.ReadKey();
}
不建議使用這種方式,這樣體現(xiàn)不出異步帶來的好處,而且使用Result屬性,有可能會帶來上下文切換造成的死鎖。
下面我們來看看創(chuàng)建Task的方法。
1、如果返回值就是一個立即可以隨手得到的值,那么就用Task.FromResult()??聪旅娲a:
staticTaskintTestAsync()
//returnTask.Runint(()=
//return5;
//});
//簡便寫法
returnTask.FromResult(3);
}
2、如果是一個需要休息一會的任務(比如下載失敗則過5秒鐘后重試。主線程不休息,和Thread.Sleep不一樣),那么就用Task.Delay()。
3、Task.Factory.FromAsync()會把IAsyncResult轉(zhuǎn)換為Task,這樣APM風格的API也可以用await來調(diào)用。
4、編寫異步方法的簡化寫法。如果方法聲明為async,那么可以直接retu
溫馨提示
- 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
- 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
- 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會有圖紙預覽,若沒有圖紙預覽就沒有圖紙。
- 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
- 5. 人人文庫網(wǎng)僅提供信息存儲空間,僅對用戶上傳內(nèi)容的表現(xiàn)方式做保護處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負責。
- 6. 下載文件中如有侵權(quán)或不適當內(nèi)容,請與我們聯(lián)系,我們立即糾正。
- 7. 本站不保證下載資源的準確性、安全性和完整性, 同時也不承擔用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。
最新文檔
- 會計學堂考試試題及答案
- 快速適應新環(huán)境能力測試題及答案
- 2025年國家公務員政治理論知識考試練習題(含答案)
- 2025年《醫(yī)療器械經(jīng)營監(jiān)督管理辦法》試題及答案
- 營救人質(zhì)考試題及答案
- LG(中國)校招面試題及答案
- 大學思修試題題庫及答案
- 未來五年自動化測試設備企業(yè)數(shù)字化轉(zhuǎn)型與智慧升級戰(zhàn)略分析研究報告
- 中煤第三建設集團(貴州)有限責任公司項目部管技人員招聘參考題庫附答案
- 興業(yè)銀行2026春季校園招聘備考題庫附答案
- 2025年云南省普洱市事業(yè)單位招聘考試(833人)高頻重點提升(共500題)附帶答案詳解
- DB15-T 3677-2024 大興安嶺林區(qū)白樺樹汁采集技術規(guī)程
- 2024年《13464電腦動畫》自考復習題庫(含答案)
- 義務教育階段學生語文核心素養(yǎng)培養(yǎng)的思考與實踐
- 綜合利用1噸APT渣項目研究報告樣本
- JT-T 1495-2024 公路水運危險性較大工程專項施工方案編制審查規(guī)程
- 圓錐曲線壓軸題30題2023
- 浙江省杭州市2022-2023學年四年級上學期語文期末試卷(含答案)2
- 試模報告模板
- 《我們?yōu)槭裁匆獙W習》的主題班會
- 海岸動力學課后習題答案詳解
評論
0/150
提交評論