<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
最近在寫程式的時候,經常遇到大量需要非同步存取的情況,但是對於async和await到底怎麼寫,還不是非常明確。
class Program { static void Main(string[] args) { MyDownLoadString ds = new MyDownLoadString(); ds.DoRun(); Console.ReadKey(); } class MyDownLoadString { Stopwatch sw = new Stopwatch(); public void DoRun() { const int LargeNumber = 6000000; sw.Start(); int t1 = CountCharacters(1, "http://www.microsoft.com"); int t2 = CountCharacters(2, "http://www.illustratedcsharp.com"); CountToALargeNumber(1, LargeNumber); CountToALargeNumber(2, LargeNumber); CountToALargeNumber(3, LargeNumber); CountToALargeNumber(4, LargeNumber); Console.WriteLine("Chars in Call1:{0}",t1); Console.WriteLine("Chars in Call1:{0}",t2); } private int CountCharacters(int id, string uriString) { WebClient wc1 = new WebClient(); Console.WriteLine("Call {0} start: {1:N0}ms ", id, sw.Elapsed.TotalMilliseconds); string result = wc1.DownloadString(new Uri(uriString)); Console.WriteLine("Call {0} completed: {1:N0}ms", id, sw.Elapsed.TotalMilliseconds); return result.Length; } private void CountToALargeNumber(int id, int value) { for (long i = 0; i < value; i++) ; Console.WriteLine("End CountToALargeNumber {0} : {1:N0}ms", id, sw.Elapsed.TotalMilliseconds); } } }
結果:
Call 1 start: 1ms
Call 1 completed: 903ms
Call 2 start: 903ms
Call 2 completed: 1,355ms
End CountToALargeNumber 1 : 1,375ms
End CountToALargeNumber 2 : 1,399ms
End CountToALargeNumber 3 : 1,417ms
End CountToALargeNumber 4 : 1,435ms
Chars in Call1:161702
Chars in Call1:5164
從執行結果可以看到,同步執行的時間主要花在了兩次請求外部地址上,計算長度並不費時,用圖來表示就像下面
修改上面程式碼,如下
class MyDownLoadString { Stopwatch sw = new Stopwatch(); public void DoRun() { const int LargeNumber = 6000000; sw.Start(); // Task<int> 儲存結果物件,後面t1.Result則是獲取結果 Task<int> t1 = CountCharactersAsync(1, "http://www.microsoft.com"); Task<int> t2 = CountCharactersAsync(2, "http://www.illustratedcsharp.com"); //無需等待CountCharactersAsync執行完成 CountToALargeNumber(1, LargeNumber); CountToALargeNumber(2, LargeNumber); CountToALargeNumber(3, LargeNumber); CountToALargeNumber(4, LargeNumber); //t1.Result獲取結果 Console.WriteLine("Chars in Call1:{0}",t1.Result); Console.WriteLine("Chars in Call1:{0}",t2.Result); } private async Task<int> CountCharactersAsync(int id, string uriString) { WebClient wc = new WebClient(); Console.WriteLine("Call {0} start: {1:N0}ms ", id, sw.Elapsed.TotalMilliseconds); string result = await wc.DownloadStringTaskAsync(new Uri(uriString)); Trace.TraceInformation("Taceing Async Call {0} @time:{1:N0}ms", id, sw.Elapsed.TotalMilliseconds); Console.WriteLine("Call {0} completed: {1:N0}ms", id, sw.Elapsed.TotalMilliseconds); return result.Length; } private void CountToALargeNumber(int id, int value) { for (long i = 0; i < value; i++) ; Console.WriteLine("End CountToALargeNumber {0}: {1:N0}ms", id, sw.Elapsed.TotalMilliseconds); } }
執行結果:
Call 1 start: 2ms
Call 2 start: 253ms
End CountToALargeNumber 1: 288ms
End CountToALargeNumber 2: 359ms
End CountToALargeNumber 3: 560ms
Call 1 completed: 770ms
End CountToALargeNumber 4: 844ms
Call 2 completed: 887ms
Chars in Call1:162262
Chars in Call2:5164
修改如上面的程式碼之後,我們就可以無需等待兩次CountCharactersAsync返回結果,而是直接呼叫了下面的CountToALargeNumber,在CountCharactersAsync請求返回的時候再獲取結果。
async和await可以建立和使用非同步方法,這個特性的由三個部分組成:
舉例說明一個async/await方法:
//1.呼叫方法 static void Main(string[] args) { Task<int> t = DoSumAsync(1, 2); Console.WriteLine("結果:{0}", t.Result); Console.ReadKey(); } //2.非同步方法 public static async Task<int> DoSumAsync(int a, int b) { //3.await 表示式 int sum = await Task.Run(() => { return a + b; }); return sum; }
上面簡單舉例了什麼是非同步方法,下面就詳細學習一下:
非同步方法在完成其工作之前返回到呼叫方法,並在呼叫方法繼續執行的時候完成其工作。語法上有如下特徵:
像程式碼:
private async Task<int> CountCharactersAsync(int id, string uriString) { WebClient wc = new WebClient(); Console.WriteLine("Call {0} start: {1:N0}ms ", id, sw.Elapsed.TotalMilliseconds); string result = await wc.DownloadStringTaskAsync(new Uri(uriString)); Trace.TraceInformation("Taceing Async Call {0} @time:{1:N0}ms", id, sw.Elapsed.TotalMilliseconds); Console.WriteLine("Call {0} completed: {1:N0}ms", id, sw.Elapsed.TotalMilliseconds); return result.Length; }
詳細說明:
①async關鍵字是一個上下文關鍵字,也就是說除了做為方法(lambda和匿名函數)的修飾符之外,還可以做識別符號。
②返回型別
首先要明確“非同步方法”的三個部分,如下圖所示:
執行過程,可以參考下面的圖
有幾個注意的地方:
- 非同步執行await表達是的空閒任務
- 當await表示式執行完成之後,執行後續部分。後續本身也可能是await表示式,處理過程和上一個一致。
- 後續部分如果遇到 return 或者 方法達到末尾,將做如下的事情:
l 如果返回的型別是void,控制流就退出了
l 如果返回的型別是Task,後續部分設定Task物件的屬性並退出。
l 如果返回的型別是Task<T>,不僅要設定Task物件屬性,還要設定Task物件的Return屬性。
這個點要注意下:並不是遇到return或者達到方法末尾,就能獲取到返回值,它只是退出了。
await表示式指定了一個非同步執行的任務。語法由 await關鍵字 + 一個空閒物件(稱為任務)組成。這個任務可能是一個Task物件,也可以不是,預設情況下由該執行緒非同步執行。
一個空閒物件 指的是一個awaitable型別的範例,awaitable型別是指包含了GetAwaiter方法的型別,方法沒有引數,返回一個稱為awaiter型別的物件。
一個awaiter物件包含了如下成員:
一般情況下我們不需要自己構建一個awaiter物件,使用.net 自己的Task就可以了。最簡單的方法就是使用Task.Run()來返回一個Task物件。關於Task.Run()有一個非常重要的點,他將在不同的執行緒上執行你的方法。
先看下面這個例子,直接在非同步方法內部使用了try..catch。
static void Main(string[] args) { Task t = BadAsync(); t.Wait(); Console.WriteLine("Task Status: {0}", t.Status ); Console.WriteLine("Task IsFaulted: {0}", t.IsFaulted ); Console.WriteLine("Please enter a key to exit!"); Console.ReadKey(); } static async Task BadAsync() { try { await Task.Run(() => { throw new Exception(); }); } catch { Console.WriteLine("Exception in BadAsync"); } }
執行結果:
Exception in BadAsync
Task Status: RanToCompletion
Task IsFaulted: False
Please enter a key to exit!
從結果可以看到,雖然在非同步方法內部進行了try..catch,並且也catch到了異常,但是對於呼叫函數,返回的Task狀態依然為 RanToCompletion 。
為什麼這個亞子?,原因如下:
對於單個Task ,可以通過task物件的wait()方法來進行等待。
Task<int> t = CountCharactersAsync("http://www.163.com"); t.Wait();
對於多個Task,可以使用WaitAll()或者waitAny()方法,進行同步。
WaitAll是等待所以的任務完成才繼續操作
Task<int> t1 = CountCharactersAsync(1, "http://www.163.com"); Task<int> t2 = CountCharactersAsync(2, "http://www.microsoft.com"); Task<int>[] tasks = new Task<int>[] { t1, t2 }; Task.WaitAll(tasks);
WaitAny是隻要一個完成就可以繼續操作
Task<int> t1 = CountCharactersAsync(1, "http://www.163.com"); Task<int> t2 = CountCharactersAsync(2, "http://www.microsoft.com"); Task<int>[] tasks = new Task<int>[] { t1, t2 }; Task.WaitAny(tasks);
上面說明了如何在“呼叫方法”中,同步等待Task的完成。 但是有時候,我們在一個非同步方法中也會存在多個任務,想要讓它們通過await表示式等待。我們可以通過Task.WhenAll() 和 Task.WhenAny() 方法實現。 這兩個方法稱為組合子(combinator)。
private async Task<int> CountCharactersAsync(string site1, string site2) { WebClient wc1 = new WebClient(); WebClient wc2 = new WebClient(); Task<string> t1 = wc1.DownloadStringTaskAsync(new Uri(site1)); Task<string> t2 = wc2.DownloadStringTaskAsync(new Uri(site2)); List<Task<string>> tasks = new List<Task<string>>(); tasks.Add(t1); tasks.Add(t2); //組合子 await Task.WhenAll(tasks); //await Task.WhenAny(tasks); Console.WriteLine(" CCA: T1 {0} Finished", t1.IsCompleted ? "" : "Not"); Console.WriteLine(" CCA: T2 {0} Finished", t2.IsCompleted ? "" : "Not"); return t1.IsCompleted? t1.Result.Length: t2.Result.Length; }
一般我們都使用Thread.Sleep(xxxx) 進行執行緒的延時,但是 Thread.Sleep會阻塞執行緒。而Task.Delay則不會阻塞執行緒,執行緒可以繼續處理其他的工作。
class Simple { Stopwatch sw = new Stopwatch(); public void DoRun() { Console.WriteLine("Caller: Before call"); ShowDelayAsync(); Console.WriteLine("Caller: After call"); } private async void ShowDelayAsync() { sw.Start(); Console.WriteLine(" Before Delay: {0} ", sw.Elapsed.Milliseconds ); await Task.Delay(1000); Console.WriteLine(" After Delay: {0} ", sw.Elapsed.Milliseconds); } }
到此這篇關於C#使用async和await實現非同步程式設計的文章就介紹到這了。希望對大家的學習有所幫助,也希望大家多多支援it145.com。
相關文章
<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
综合看Anker超能充系列的性价比很高,并且与不仅和iPhone12/苹果<em>Mac</em>Book很配,而且适合多设备充电需求的日常使用或差旅场景,不管是安卓还是Switch同样也能用得上它,希望这次分享能给准备购入充电器的小伙伴们有所
2021-06-01 09:31:42
除了L4WUDU与吴亦凡已经多次共事,成为了明面上的厂牌成员,吴亦凡还曾带领20XXCLUB全队参加2020年的一场音乐节,这也是20XXCLUB首次全员合照,王嗣尧Turbo、陈彦希Regi、<em>Mac</em> Ova Seas、林渝植等人全部出场。然而让
2021-06-01 09:31:34
目前应用IPFS的机构:1 谷歌<em>浏览器</em>支持IPFS分布式协议 2 万维网 (历史档案博物馆)数据库 3 火狐<em>浏览器</em>支持 IPFS分布式协议 4 EOS 等数字货币数据存储 5 美国国会图书馆,历史资料永久保存在 IPFS 6 加
2021-06-01 09:31:24
开拓者的车机是兼容苹果和<em>安卓</em>,虽然我不怎么用,但确实兼顾了我家人的很多需求:副驾的门板还配有解锁开关,有的时候老婆开车,下车的时候偶尔会忘记解锁,我在副驾驶可以自己开门:第二排设计很好,不仅配置了一个很大的
2021-06-01 09:30:48
不仅是<em>安卓</em>手机,苹果手机的降价力度也是前所未有了,iPhone12也“跳水价”了,发布价是6799元,如今已经跌至5308元,降价幅度超过1400元,最新定价确认了。iPhone12是苹果首款5G手机,同时也是全球首款5nm芯片的智能机,它
2021-06-01 09:30:45