一文說通C#中的非同步編程

来源:https://www.cnblogs.com/tiger-wang/archive/2020/07/22/13357981.html
-Advertisement-
Play Games

天天寫,不一定就明白。 又及,前兩天看了一個關於同步方法中調用非同步方法的文章,裡面有些概念不太正確,所以整理了這個文章。 一、同步和非同步。 先說同步。 同步概念大家都很熟悉。在非同步概念出來之前,我們的代碼都是按同步的方式寫的。簡單來說,就是程式嚴格按照代碼的邏輯次序,一行一行執行。 看一段代碼: p ...


天天寫,不一定就明白。

又及,前兩天看了一個關於同步方法中調用非同步方法的文章,裡面有些概念不太正確,所以整理了這個文章。

一、同步和非同步。

先說同步。

同步概念大家都很熟悉。在非同步概念出來之前,我們的代碼都是按同步的方式寫的。簡單來說,就是程式嚴格按照代碼的邏輯次序,一行一行執行。

看一段代碼:

public static void Main(string[] args)
{
    Console.WriteLine("Syc proccess - start");

    Console.WriteLine("Syc proccess - enter Func1");
    func1();
    Console.WriteLine("Syc proccess - out Func1");

    Console.WriteLine("Syc proccess - enter Func2");
    func2();
    Console.WriteLine("Syc proccess - out Func2");

    Console.WriteLine("Syc proccess - enter Func3");
    func3();
    Console.WriteLine("Syc proccess - out Func3");

    Console.WriteLine("Syc proccess - done");
}

private static void func1()
{
    Console.WriteLine("Func1 proccess - start");
    Thread.Sleep(1000);
    Console.WriteLine("Func1 proccess - end");
}

private static void func2()
{
    Console.WriteLine("Func2 proccess - start");
    Thread.Sleep(3000);
    Console.WriteLine("Func2 proccess - end");
}

private static void func3()
{
    Console.WriteLine("Func3 proccess - start");
    Thread.Sleep(5000);
    Console.WriteLine("Func3 proccess - end");
}

這是一段簡單的通常意義上的代碼,程式按代碼的次序同步執行,看結果:

Syc proccess - start
Syc proccess - enter Func1
Func1 proccess - start
Func1 proccess - end
Syc proccess - out Func1
Syc proccess - enter Func2
Func2 proccess - start
Func2 proccess - end
Syc proccess - out Func2
Syc proccess - enter Func3
Func3 proccess - start
Func3 proccess - end
Syc proccess - out Func3
Syc proccess - done

沒有任何意外。

    為了防止不提供原網址的轉載,特在這裡加上原文鏈接:https://www.cnblogs.com/tiger-wang/p/13357981.html

那非同步呢?

非同步,來自於對同步處理的改良和優化。

應用中,經常會有對於文件或網路、資料庫的IO操作。這些操作因為IO軟硬體的原因,需要消耗很多時間,但通常情況下CPU計算量並不大。在同步的代碼中,這個過程會被阻塞。直白的說法就是這一行代碼沒執行完成,程式就得等著,等完成後再執行下一行代碼,而這個等待的時間中,CPU資源就被浪費了,閑著了,什麼也沒做。(當然,操作系統會調度CPU乾別的,這兒不抬杠。)

非同步編程模型和規範因此出現了,通過某種機制,讓程式在等著IO的過程中,繼續做點別的事,等IO的過程完成了,再回來處理IO的內容。這樣CPU也沒閑著,在等IO的過程中多做了點事。反映到用戶端,就感覺程式更快了,用時更短了。

下麵重點說一下非同步編程相關的內容。

二、非同步編程

C#中,非同步編程,一個核心,兩個關鍵字。

一個核心是指TaskTask<T>對象,而兩個關鍵字,就是asyncawait

從各種渠道給出的非同步編程,都是下麵的方式:

async Task function()
{
  /* your code here */
}

然後調用的方式:

await function();

是這樣的嗎?嗯,圖樣圖森破~~~

我們來看代碼:

static async Task Main(string[] args)
{
    Console.WriteLine("Async proccess - start");

    Console.WriteLine("Async proccess - enter Func1");
    await func1();
    Console.WriteLine("Async proccess - out Func1");

    Console.WriteLine("Async proccess - enter Func2");
    await func2();
    Console.WriteLine("Async proccess - out Func2");

    Console.WriteLine("Async proccess - enter Func3");
    await func3();
    Console.WriteLine("Async proccess - out Func3");

    Console.WriteLine("Async proccess - done");

    Console.WriteLine("Main proccess - done");
}

private static async Task func1()
{
    Console.WriteLine("Func1 proccess - start");
    Thread.Sleep(1000);
    Console.WriteLine("Func1 proccess - end");
}

private static async Task func2()
{
    Console.WriteLine("Func2 proccess - start");
    Thread.Sleep(3000);
    Console.WriteLine("Func2 proccess - end");
}

private static async Task func3()
{
    Console.WriteLine("Func3 proccess - start");
    Thread.Sleep(5000);
    Console.WriteLine("Func3 proccess - end");
}

跑一下結果:

Async proccess - start
Async proccess - enter Func1
Func1 proccess - start
Func1 proccess - end
Async proccess - out Func1
Async proccess - enter Func2
Func2 proccess - start
Func2 proccess - end
Async proccess - out Func2
Async proccess - enter Func3
Func3 proccess - start
Func3 proccess - end
Async proccess - out Func3
Async proccess - done
Main proccess - done

咦?這個好像跟同步代碼的執行結果沒什麼區別啊?

嗯,完全正確。上面這個代碼,真的是同步執行的。

這是非同步編程的第一個容易錯誤的理解:asyncawait的配對。

三、async和await的配對

在非同步編程的規範中,async修飾的方法,僅僅表示這個方法在內部有可能採用非同步的方式執行,CPU在執行這個方法時,會放到一個新的線程中執行。

那這個方法,最終是否採用非同步執行,不決定於是否用await方式調用這個方法,而決定於這個方法內部,是否有await方式的調用。

看代碼,很容易理解:

private static async Task func1()
{
    Console.WriteLine("Func1 proccess - start");
    Thread.Sleep(1000);
    Console.WriteLine("Func1 proccess - end");
}

這個方法,因為方法內部沒有await調用,所以這個方法永遠會以同步方式執行,不管你調用這個方法時,有沒有await

而下麵這個代碼:

private static async Task func1()
{
    Console.WriteLine("Func1 proccess - start");
    await Task.Run(() => Thread.Sleep(1000));
    Console.WriteLine("Func1 proccess - end");
}

因為這個方法里有await調用,所以這個方法不管你以什麼方式調用,有沒有await,都是非同步執行的。

看代碼:

static async Task Main(string[] args)
{
    Console.WriteLine("Async proccess - start");

    Console.WriteLine("Async proccess - enter Func1");
    func1();
    Console.WriteLine("Async proccess - out Func1");

    Console.WriteLine("Async proccess - enter Func2");
    func2();
    Console.WriteLine("Async proccess - out Func2");

    Console.WriteLine("Async proccess - enter Func3");
    func3();
    Console.WriteLine("Async proccess - out Func3");

    Console.WriteLine("Async proccess - done");

    Console.WriteLine("Main proccess - done");

    Console.ReadKey();
}

private static async Task func1()
{
    Console.WriteLine("Func1 proccess - start");
    await Task.Run(() => Thread.Sleep(1000));
    Console.WriteLine("Func1 proccess - end");
}

private static async Task func2()
{
    Console.WriteLine("Func2 proccess - start");
    await Task.Run(() => Thread.Sleep(3000));
    Console.WriteLine("Func2 proccess - end");
}

private static async Task func3()
{
    Console.WriteLine("Func3 proccess - start");
    await Task.Run(() => Thread.Sleep(5000));
    Console.WriteLine("Func3 proccess - end");
}

輸出結果:

Async proccess - start
Async proccess - enter Func1
Func1 proccess - start
Async proccess - out Func1
Async proccess - enter Func2
Func2 proccess - start
Async proccess - out Func2
Async proccess - enter Func3
Func3 proccess - start
Async proccess - out Func3
Async proccess - done
Main proccess - done
Func1 proccess - end
Func2 proccess - end
Func3 proccess - end

結果中,在長時間運行Thread.Sleep的時候,跳出去往下執行了,是非同步。

又有問題來了:不是說非同步調用要用await嗎?

我們把await加到調用方法的前邊,試一下:

	   

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 今天總結一下,調用第三方介面地址,以POST方式進行HTTP請求,並且入參為JSON格式方法: 引用:using System.Text;using System.Net;using System.IO; string url="http://";//第三方介面地址 paramStr //json字 ...
  • 1.首先使用VS創建WebAPI項目 (這裡有個幫助類,將此幫助類複製到項目里,有興趣可以學著寫) //文件上傳下載,導入導出輔助類 public class APIFileHelp { //此為限制文件格式 public string[] ExtentsfileName = new string[ ...
  • 今天總結一下 關於XML字元串轉DataTable 方法: 引用:using System.Xml; using Newtonsoft.Json;using System.Data; using System.Collections; 首先,定義一個xml字元串來接收傳過來的數據, string x ...
  • 今天總結一下關於DataTable,XML轉JSON的方法: 首先需要引入命名空間: using Newtonsoft.Json 1 public string DataTableToJsonWithStringBuilder(DataTable table) 2 { 3 var jsonStrin ...
  • 最近在做調用第三方介面,要求入參AES加密,並且秘鑰為16位的長度,在此記錄一下。 首先引用命名空間: using System.IO; using System.Text; using System.Security.Cryptography; 1 /// <summary> 2 /// AES加 ...
  • 上篇我們完成了數據源列表展示功能(還未測試)。 本篇我們來新增數據源,並查看列表展示功能。 接上篇: 二、數據源管理功能開發 2、新增數據源 我們用模態對話框來完成數據源的新增,效果如下圖: 我們分兩部分講解:展示 和 邏輯。 展示: 我們用的前端UI是基於bootstrap的,因此bootstra ...
  • Tips:本篇已加入系列文章閱讀目錄,可點擊查看更多相關文章。 前言 Docker 是一個開源的應用容器引擎,它十分火熱,如今幾乎成為了後端開發人員必須掌握的一項技能。即使你在生產環境中可能用不上它,就算把它當作一個輔助開發的工具來使用,也是非常方便的。本文就介紹一下.Net Core應用在Dock ...
  • 1.原因 載入的時候沒有調取 AssemblyLoadContext.Default 2.解決方案: 在程式啟動的時候,手動調用 /// <summary> /// Interaction logic for App.xaml /// </summary> public partial class ...
一周排行
    -Advertisement-
    Play Games
  • Dapr Outbox 是1.12中的功能。 本文只介紹Dapr Outbox 執行流程,Dapr Outbox基本用法請閱讀官方文檔 。本文中appID=order-processor,topic=orders 本文前提知識:熟悉Dapr狀態管理、Dapr發佈訂閱和Outbox 模式。 Outbo ...
  • 引言 在前幾章我們深度講解了單元測試和集成測試的基礎知識,這一章我們來講解一下代碼覆蓋率,代碼覆蓋率是單元測試運行的度量值,覆蓋率通常以百分比表示,用於衡量代碼被測試覆蓋的程度,幫助開發人員評估測試用例的質量和代碼的健壯性。常見的覆蓋率包括語句覆蓋率(Line Coverage)、分支覆蓋率(Bra ...
  • 前言 本文介紹瞭如何使用S7.NET庫實現對西門子PLC DB塊數據的讀寫,記錄了使用電腦模擬,模擬PLC,自至完成測試的詳細流程,並重點介紹了在這個過程中的易錯點,供參考。 用到的軟體: 1.Windows環境下鏈路層網路訪問的行業標準工具(WinPcap_4_1_3.exe)下載鏈接:http ...
  • 從依賴倒置原則(Dependency Inversion Principle, DIP)到控制反轉(Inversion of Control, IoC)再到依賴註入(Dependency Injection, DI)的演進過程,我們可以理解為一種逐步抽象和解耦的設計思想。這種思想在C#等面向對象的編 ...
  • 關於Python中的私有屬性和私有方法 Python對於類的成員沒有嚴格的訪問控制限制,這與其他面相對對象語言有區別。關於私有屬性和私有方法,有如下要點: 1、通常我們約定,兩個下劃線開頭的屬性是私有的(private)。其他為公共的(public); 2、類內部可以訪問私有屬性(方法); 3、類外 ...
  • C++ 訪問說明符 訪問說明符是 C++ 中控制類成員(屬性和方法)可訪問性的關鍵字。它們用於封裝類數據並保護其免受意外修改或濫用。 三種訪問說明符: public:允許從類外部的任何地方訪問成員。 private:僅允許在類內部訪問成員。 protected:允許在類內部及其派生類中訪問成員。 示 ...
  • 寫這個隨筆說一下C++的static_cast和dynamic_cast用在子類與父類的指針轉換時的一些事宜。首先,【static_cast,dynamic_cast】【父類指針,子類指針】,兩兩一組,共有4種組合:用 static_cast 父類轉子類、用 static_cast 子類轉父類、使用 ...
  • /******************************************************************************************************** * * * 設計雙向鏈表的介面 * * * * Copyright (c) 2023-2 ...
  • 相信接觸過spring做開發的小伙伴們一定使用過@ComponentScan註解 @ComponentScan("com.wangm.lifecycle") public class AppConfig { } @ComponentScan指定basePackage,將包下的類按照一定規則註冊成Be ...
  • 操作系統 :CentOS 7.6_x64 opensips版本: 2.4.9 python版本:2.7.5 python作為腳本語言,使用起來很方便,查了下opensips的文檔,支持使用python腳本寫邏輯代碼。今天整理下CentOS7環境下opensips2.4.9的python模塊筆記及使用 ...