C#5.0中的異步編程關(guān)鍵字async和await
一、Asynchronous methods 異步方法
.NET 4.5 的推出,對(duì)于C#又有了新特性的增加——就是C#5.0中async和await兩個(gè)關(guān)鍵字,這兩個(gè)關(guān)鍵字簡(jiǎn)化了異步編程。
- 使用async修飾的方法被稱為異步方法,這個(gè)方法調(diào)用時(shí)應(yīng)該在前面加上await。
- 異步方法命名應(yīng)該以Async結(jié)尾,這樣大家知道調(diào)用的時(shí)候使用await。
async和await關(guān)鍵字只是編譯器的功能,編譯器最終會(huì)用Task類創(chuàng)建代碼。
1、創(chuàng)建返回任務(wù)的異步方法
建立一個(gè)同步方法Greeting,該方法在等待一段時(shí)間后,返回一個(gè)字符串。
private string Greeting(int delay, string name) { System.Threading.Thread.Sleep(delay); return string.Format("Hello, {0}.", name); }
定義一個(gè)方法GreetingAsync,可以使方法異步化,其傳入的參數(shù)不做強(qiáng)制要求。
異步方法,返回類型必須為Task、Task或void。不能作為程序的入口點(diǎn),即Main方法不能使用async修飾符。
基于任務(wù)的異步模式指定,并返回一個(gè)任務(wù)。
注意,該方法返回的是Task,定義了一個(gè)返回字符串的任務(wù),與同步方法返回值一致。
private Task<string> GreetingAsync(string name, int delay = 3000) { return Task.Run<string>(() => { return Greeting(delay, name); }); }
2、調(diào)用異步方法
可以使用await關(guān)鍵字調(diào)用返回任務(wù)的異步方法GreetingAsync。
注意:await修飾符只能用于返回Task或者Task的方法。
并且使用await關(guān)鍵字的方法,這里是CallerWithAsync(),必須要用async關(guān)鍵字修飾符聲明。
在GreetingAsync方法完成前,被async關(guān)鍵字修飾的方法內(nèi)await關(guān)鍵字后面的代碼不會(huì)繼續(xù)執(zhí)行。
但是,啟動(dòng)被async關(guān)鍵字修飾的方法的線程可以被重用,而沒有被阻塞。
public async void CallerWithAsync() { string result = await GreetingAsync("Nigel", 2000); Console.WriteLine(result); }
3、簡(jiǎn)單實(shí)例
void Main() { DisplayValue(); System.Diagnostics.Debug.WriteLine("MyClass() End."); } public async void DisplayValue() { double result = await GetValueAsync(1234.5, 1.01);//此處會(huì)開新線程處理GetValueAsync任務(wù),然后方法馬上返回。這之后的所有代碼都會(huì)被封裝成委托,在GetValueAsync任務(wù)完成時(shí)調(diào)用 System.Diagnostics.Debug.WriteLine("Value is : " + result); } public Task<double> GetValueAsync(double num1, double num2) { return Task.Run(() => { for (int i = 0; i < 1000000; i++) { num1 = num1 / num2; } return num1; }); }
上面在MyClass的構(gòu)造函數(shù)里調(diào)用了async關(guān)鍵字標(biāo)記的異步方法DisplayValue(),DisplayValue()方法里執(zhí)行了一個(gè)await關(guān)鍵字標(biāo)記的異步任務(wù)GetValueAsync(),這個(gè)異步任務(wù)必須是以Task或者Task作為返回值的。
而我們也看到,異步任務(wù)執(zhí)行完成時(shí)實(shí)際返回的類型是void或者TResult,DisplayValue()方法里await GetValueAsync()之后的所有代碼都會(huì)在異步任務(wù)完成時(shí)才會(huì)執(zhí)行。
DisplayValue()方法實(shí)際執(zhí)行的代碼如下:
public void DisplayValue() { System.Runtime.CompilerServices.TaskAwaiter<double> awaiter = GetValueAsync(1234.5, 1.01).GetAwaiter(); awaiter.OnCompleted(() => { double result = awaiter.GetResult(); System.Diagnostics.Debug.WriteLine("Value is : " + result); }); }
可以看到,async和await關(guān)鍵字只是把上面的代碼變得更簡(jiǎn)單易懂而已。
程序的輸出如下:
MyClass() End.
Value is : 2.47032822920623E-322
4、使用async 和await定義異步方法不會(huì)創(chuàng)建新線程, 它運(yùn)行在現(xiàn)有線程上執(zhí)行多個(gè)任務(wù)。
// 使用C# 5.0中提供的async 和await關(guān)鍵字來定義異步方法 // 從代碼中可以看出C#5.0 中定義異步方法就像定義同步方法一樣簡(jiǎn)單。 private async Task<long> AccessWebAsync() { MemoryStream content = new MemoryStream(); // 對(duì)MSDN發(fā)起一個(gè)Web請(qǐng)求 HttpWebRequest webRequest = WebRequest.Create("http://msdn.microsoft.com/zh-cn/") as HttpWebRequest; if (webRequest != null) { // 返回回復(fù)結(jié)果 using (WebResponse response = await webRequest.GetResponseAsync()) { using (Stream responseStream = response.GetResponseStream()) { await responseStream.CopyToAsync(content); } } } txbAsynMethodID.Text = Thread.CurrentThread.ManagedThreadId.ToString(); return content.Length; }
運(yùn)行結(jié)果如下:
三、async和await關(guān)鍵字剖析
我們對(duì)比下上面使用async和await關(guān)鍵字來實(shí)現(xiàn)異步編程的代碼和在第二部分的同步代碼,有沒有發(fā)現(xiàn)使用async和await關(guān)鍵字的異步實(shí)現(xiàn)和同步代碼的實(shí)現(xiàn)很像,只是異步實(shí)現(xiàn)中多了async和await關(guān)鍵字和調(diào)用的方法都多了async后綴而已。
正是因?yàn)樗麄兊膶?shí)現(xiàn)很像,所以我在第四部分才命名為使用async和await使異步編程更簡(jiǎn)單,就像我們?cè)趯懲酱a一樣,并且代碼的coding思路也是和同步代碼一樣,這樣就避免考慮在APM中委托的回調(diào)等復(fù)雜的問題,以及在EAP中考慮各種事件的定義。
下面再分享下幾個(gè)關(guān)于async和await常問的問題
- 問題一:是不是寫了async關(guān)鍵字的方法就代表該方法是異步方法,不會(huì)堵塞線程呢?
答: 不是的,對(duì)于只標(biāo)識(shí)async關(guān)鍵字的(指在方法內(nèi)沒有出現(xiàn)await關(guān)鍵字)的方法,調(diào)用線程會(huì)把該方法當(dāng)成同步方法一樣執(zhí)行,所以然而會(huì)堵塞GUI線程,只有當(dāng)async和await關(guān)鍵字同時(shí)出現(xiàn),該方法才被轉(zhuǎn)換為異步方法處理。
- 問題二:“async”關(guān)鍵字會(huì)導(dǎo)致調(diào)用方法用線程池線程運(yùn)行嗎?
答: 不會(huì),被async關(guān)鍵字標(biāo)識(shí)的方法不會(huì)影響方法是同步還是異步運(yùn)行并完成,而是,它使方法可被分割成多個(gè)片段,其中一些片段可能異步運(yùn)行,這樣這個(gè)方法可能異步完成。這些片段界限就出現(xiàn)在方法內(nèi)部顯示使用”await”關(guān)鍵字的位置處。所以,如果在標(biāo)記了”async”的方法中沒有顯示使用”await”,那么該方法只有一個(gè)片段,并且將以同步方式運(yùn)行并完成。在await關(guān)鍵字出現(xiàn)的前面部分代碼和后面部分代碼都是同步執(zhí)行的(即在調(diào)用線程上執(zhí)行的,也就是GUI線程,所以不存在跨線程訪問控件的問題),await關(guān)鍵處的代碼片段是在線程池線程上執(zhí)行??偨Y(jié)為——使用async和await關(guān)鍵字實(shí)現(xiàn)的異步方法,此時(shí)的異步方法被分成了多個(gè)代碼片段去執(zhí)行的,而不是像之前的異步編程模型(APM)和EAP那樣,使用線程池線程去執(zhí)行一整個(gè)方法。
到此這篇關(guān)于C#5.0異步編程之a(chǎn)sync和await的文章就介紹到這了。希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
C#基礎(chǔ):Dispose()、Close()、Finalize()的區(qū)別詳解
本篇文章是對(duì)c#中的Dispose()、Close()、Finalize()的區(qū)別進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-05-05C#數(shù)據(jù)結(jié)構(gòu)與算法揭秘四 雙向鏈表
上節(jié)說過這節(jié)會(huì)講雙向鏈表,環(huán)形鏈表和應(yīng)用舉例,我們開始吧!?。?!2012-11-11