眾所周知C#提供Async和Await關鍵字來實現異步編程。在本文中,我們將共同探討並介紹什么是Async 和 Await,以及如何在C#中使用Async 和 Await。
同樣本文的內容也大多是翻譯的,只不過加上了自己的理解進行了相關知識點的補充,如果你認為自己的英文水平還不錯,大可直接跳轉到文章末尾查看原文鏈接進行閱讀。
寫在前面
自從C# 5.0時代引入async和await關鍵字后,異步編程就變得流行起來。尤其在現在的.NET Core時代,如果你的代碼中沒有出現async或者await關鍵字,都會讓人感覺到很奇怪。
想象一下當我們在處理UI和按鈕單擊時,我們需要運行一個長時間運行的方法,比如讀取一個大文件或其他需要很長時間的任務,在這種情況下,整個應用程序必須等待這個長時間運行的任務完成才算完成整個任務。
換句話說,如果同步應用程序中的任何進程被阻塞,則整個應用程序將被阻塞,我們的應用程序將停止響應,直到整個任務完成。
在這種情況下,異步編程將非常有用。通過使用異步編程,應用程序可以繼續進行不依賴於整個任務完成的其他工作。
在Async 和 await關鍵字的幫助下,使得異步編程變得很簡單,而且我們將獲得傳統異步編程的所有好處。
實例講解
假設我們分別使用了兩種方法,即Method 1和Method 2,這兩種方法不相互依賴,而Method 1需要很長時間才能完成它的任務。在同步編程中,它將執行第一個Method 1,並等待該方法的完成,然后執行Method 2。因此,這將是一個時間密集型的過程,即使這兩種方法並不相互依賴。
我們可以使用簡單的多線程編程並行運行所有方法,但是它會阻塞UI並等待完成所有任務。要解決這個問題,我們必須在傳統編程中編寫很多的代碼,但是現在我們有了Async 和 await關鍵字,那么我們將通過書寫很少的並且簡潔的代碼來解決這個問題。
此外,我們還將看到更多的示例,如果任何第三個方法(如Method 3)都依賴於Method 1,那么它將在Wait關鍵字的幫助下等待Method 1的完成。
Async 和 await是代碼標記,它標記代碼位置為任務完成后控件應該恢復的位置。
下面讓我們舉幾個例子來更好進行理解吧
C#中Async 和 await關鍵字的示例
我們將采用控制台應用程序進行演示。
第一個例子
在這個例子中,我們將采取兩個不相互依賴的方法。
class Program { static void Main(string[] args) { Method1(); Method2(); Console.ReadKey(); } public static async Task Method1() { await Task.Run(() => { for (int i = 0; i < 100; i++) { Console.WriteLine(" Method 1"); } }); } public static void Method2() { for (int i = 0; i < 25; i++) { Console.WriteLine(" Method 2"); } } }
在上面給出的代碼中,Method 1和Method 2不相互依賴,我們是從主方法調用的。
在這里,我們可以清楚地看到,方法1和方法2並不是在等待對方完成。
輸出
現在來看第二個例子,假設我們有Method 3,它依賴於Method 1
第二個例子
在本例中,Method 1將總長度作為整數值返回,我們在Method 3中以長度的形式傳遞一個參數,它來自Method 1。
在這里,在傳遞Method 3中的參數之前,我們必須使用AWAIT關鍵字,為此,我們必須使用調用方法中的async 關鍵字。
在控制台應用程序的Main方法中,因為不能使用async關鍵字而不能使用await 關鍵字,因為它會給出下面給出的錯誤。(但是如果你使用的是C#7.1及以上的方法是不會有問題的,因為C#7.1及以上的語法支持Mian方法前加async)
我們將創建一個新的方法,作為CallMethod,在這個方法中,我們將調用我們的所有方法,分別為Method 1、Method 2和Method 3。
class Program { static void Main(string[] args) { callMethod(); Console.ReadKey(); } public static async void callMethod() { Task<int> task = Method1(); Method2(); int count = await task; Method3(count); } public static async Task<int> Method1() { int count = 0; await Task.Run(() => { for (int i = 0; i < 100; i++) { Console.WriteLine(" Method 1"); count += 1; } }); return count; } public static void Method2() { for (int i = 0; i < 25; i++) { Console.WriteLine(" Method 2"); } } public static void Method3(int count) { Console.WriteLine("Total count is " + count); } }
在上面給出的代碼中,Method 3需要一個參數,即Method 1的返回類型。在這里,await關鍵字對於等待Method 1任務的完成起着至關重要的作用。
輸出
第三個例子
.NET Framework4.5中有一些支持API,Windows運行時包含支持異步編程的方法。
在Async 和 await關鍵字的幫助下,我們可以在實時項目中使用所有這些,以便更快地執行任務。
包含異步方法的API有HttpClient, SyndicationClient, StorageFile, StreamWriter, StreamReader, XmlReader, MediaCapture, BitmapEncoder, BitmapDecoder 等。
在本例中,我們將異步讀取大型文本文件中的所有字符,並獲取所有字符的總長度。
class Program { static void Main() { Task task = new Task(CallMethod); task.Start(); task.Wait(); Console.ReadLine(); } static async void CallMethod() { string filePath = "E:\\sampleFile.txt"; Task<int> task = ReadFile(filePath); Console.WriteLine(" Other Work 1"); Console.WriteLine(" Other Work 2"); Console.WriteLine(" Other Work 3"); int length = await task; Console.WriteLine(" Total length: " + length); Console.WriteLine(" After work 1"); Console.WriteLine(" After work 2"); } static async Task<int> ReadFile(string file) { int length = 0; Console.WriteLine(" File reading is stating"); using (StreamReader reader = new StreamReader(file)) { // Reads all characters from the current position to the end of the stream asynchronously // and returns them as one string. string s = await reader.ReadToEndAsync(); length = s.Length; } Console.WriteLine(" File reading is completed"); return length; } }
在上面給出的代碼中,我們調用ReadFile方法來讀取文本文件的內容,並獲取文本文件中總字符的長度。
在sampleText.txt中,文件包含了太多的字符,因此讀取所有字符需要很長時間。
在這里,我們使用異步編程從文件中讀取所有內容,所以它不會等待從這個方法獲得一個返回值並執行其他代碼行,但是它必須等待下面給出的代碼行,因為我們使用的是等待關鍵字,我們將對下面給出的代碼行使用返回值。
int length = await task; Console.WriteLine(" Total length: " + length);
隨后,將按順序執行其他代碼行。
Console.WriteLine(" After work 1"); Console.WriteLine(" After work 2");
輸出
最后
在這里,我們必須了解非常重要的一點,如果我們沒有使用await 關鍵字,那么該方法就作為一個同步方法。編譯器將向我們顯示警告,但不會顯示任何錯誤。
像上面這種簡單的方式一樣,我們可以在C#代碼中使用async 和await關鍵字來愉快的進行異步編程了。
最后的最后感謝大家的閱讀!
本文大部分內容翻譯自:https://www.c-sharpcorner.com/article/async-and-await-in-c-sharp/