關於C#中async/await的用法

2023-02-08 18:01:22

一直對c#中async/await的用法模模糊糊,不太清晰,今天寫了一下Demo徹底明確一下async/await的用法,以免因為對其不瞭解而對後期的業務產生影響(比如事務導致的鎖表等等)。

1. 首先,async/await 成對出現才有意義。其意義在於可以等待非同步操作完成後繼續順序執行,而不是非同步操作還沒處理完成主執行緒就進行了下一步。

    假設,我們現在要模擬簡單的下載場景,首先使用者點選下載,那麼就呼叫DownloadHandle方法(非同步)進行下載,然後通知使用者下載完成。其使用 async/await  的區別如下:

    (1)使用 async/await 的情況:

internal class Program
    {
        static void Main(string[] args)
        {
            DownloadHandle();
            Console.ReadLine();
        }
        /// <summary>
        /// 正常使用async/await時,符合正常的業務邏輯:
        /// 1. 通知使用者下載開始
        /// 2. 非同步下載
        /// 3. 等待非同步下載完成後給使用者提示下載完成
        /// </summary>
        public static async void DownloadHandle()
        {
            Console.WriteLine("下載開始!->主執行緒ID:" + Thread.CurrentThread.ManagedThreadId);
            await Download();
            Console.WriteLine("下載完成!->主執行緒ID:" + Thread.CurrentThread.ManagedThreadId);

        }
        /// <summary>
        /// 下載
        /// </summary>
        /// <returns></returns>
        public static Task Download()
        {
            return Task.Run(() =>
            {
                Console.WriteLine("下載執行緒ID:->" + Thread.CurrentThread.ManagedThreadId);
                Console.WriteLine("10%");
                Console.WriteLine("30%");
                Console.WriteLine("50%");
                Console.WriteLine("60%");
                Console.WriteLine("80%");
                Console.WriteLine("99%");
                Console.WriteLine("100%");
            });
        }
    }

結果如下:

 

 

 可以看到,即時下載使用了非同步(執行緒ID不同也表明了當前使用了非同步),業務邏輯最終還是按照我們的需求,按順序正序執行了。

(2)不使用async/await的情況:

internal class Program
    {
        static void Main(string[] args)
        {
            DownloadHandle();
            Console.ReadLine();
        }
        /// <summary>
        /// 不適用async/await時,則程式碼執行順序時混亂的,不符合業務邏輯:
        /// 1. 通知使用者下載開始
        /// 2. 提示下載完成
        /// 3. 開始下載
        /// </summary>
        public static void DownloadHandle()
        {
            Console.WriteLine("下載開始!->主執行緒ID:" + Thread.CurrentThread.ManagedThreadId);
            Download();
            Console.WriteLine("下載完成!->主執行緒ID:" + Thread.CurrentThread.ManagedThreadId);

        }
        /// <summary>
        /// 下載
        /// </summary>
        /// <returns></returns>
        public static Task Download()
        {
            return Task.Run(() =>
            {
                Console.WriteLine("下載執行緒ID:->" + Thread.CurrentThread.ManagedThreadId);
                Console.WriteLine("10%");
                Console.WriteLine("30%");
                Console.WriteLine("50%");
                Console.WriteLine("60%");
                Console.WriteLine("80%");
                Console.WriteLine("99%");
                Console.WriteLine("100%");
            });
        }
    }

結果如下:

 

 

 可以看到,程式碼執行順序混亂了,「下載完成」 跑到了 「下載執行緒ID」 前面去了,完全沒有按照我們預期的順序執行。

2. 如果可以await的方法不進行await,那將會怎樣呢?

(1)如果被呼叫的非同步方法內部使用了Task.Run,那結果可參考我們1中進行講述的結果。開發者可根據實際需要來進行呼叫,如果非同步方法的呼叫結果與其上下文邏輯沒有嚴格的執行要求,則可以不進行await(比如記錄紀錄檔等等)。反之,則需要加await。

(2)如果被呼叫的非同步方法內部只是返回了Task.CompletedTask,即時使用了await/async實際上還是等於同步執行,如下圖。

internal class Program
    {
        static void Main(string[] args)
        {
            DownloadHandle();
            Console.ReadLine();
        }
        /// <summary>
        /// 模擬下載
        /// </summary>
        public static async void DownloadHandle()
        {
            Console.WriteLine("下載開始!->主執行緒ID:" + Thread.CurrentThread.ManagedThreadId);
            await Download();
            Console.WriteLine("下載完成!->主執行緒ID:" + Thread.CurrentThread.ManagedThreadId);

        }
        /// <summary>
        /// 下載
        /// </summary>
        /// <returns></returns>
        public static Task Download()
        {
            Console.WriteLine("下載執行緒ID:->" + Thread.CurrentThread.ManagedThreadId);
            Console.WriteLine("10%");
            Console.WriteLine("30%");
            Console.WriteLine("50%");
            Console.WriteLine("60%");
            Console.WriteLine("80%");
            Console.WriteLine("99%");
            Console.WriteLine("100%");
            return Task.CompletedTask;
        }

結果如圖:

 

 

 可以看到,即使DonwloadHandle方法使用了await/async,還是進行了同步執行,並沒有非同步效果(可從所有執行緒ID相同看出)

3. 小技巧: 非同步方法的返回值型別一般都是Task或者Task<T>型別的,當返回值為Task時(即方法的返回值型別為void),我們可以直接return Task.Run(()=>{})(以下第一段程式碼),而不必await Task.Run(()=>{})(以下第二段程式碼),這樣也可從一定程度上提高程式碼執行效率。另外,不推薦使用async 修飾void返回值,會有例外處理方面的問題(非常感謝 殘生 指教)。

/// <summary>
        /// 下載
        /// </summary>
        /// <returns></returns>
        public static Task Download()
        {
            return Task.Run(() =>
            {
                Console.WriteLine("下載執行緒ID:->" + Thread.CurrentThread.ManagedThreadId);
                Console.WriteLine("10%");
                Console.WriteLine("30%");
                Console.WriteLine("50%");
                Console.WriteLine("60%");
                Console.WriteLine("80%");
                Console.WriteLine("99%");
                Console.WriteLine("100%");
            });
        }
/// <summary>
        /// 下載
        /// </summary>
        /// <returns></returns>
        public static async Task Download()
        {
            await Task.Run(() =>
            {
                Console.WriteLine("下載執行緒ID:->" + Thread.CurrentThread.ManagedThreadId);
                Console.WriteLine("10%");
                Console.WriteLine("30%");
                Console.WriteLine("50%");
                Console.WriteLine("60%");
                Console.WriteLine("80%");
                Console.WriteLine("99%");
                Console.WriteLine("100%");
            });
        }

 

以上,只是作者的個人理解,請大神勿噴,如有錯誤,歡迎指正,謝謝!