充分發揮異步在 ASP.NET 中的強大優勢
最近幾年,非同步程式設計受到極大關注,主要是出於兩個關鍵原因:首先,它有助於提供更好的使用者體驗,因為不會阻塞UI 線程,避免了處理結束前出現UI 介面掛起。其次,它有助於大幅擴展系統,而且無需添加額外硬體。
但是,編寫合適的非同步程式碼來管理執行緒本身是項乏味的工作。雖然如此,其巨大好處讓許多新舊技術紛紛開始使用非同步程式設計。微軟自發布了 .NET 4.0以後也對其投入頗多,隨後在 .NET 4.5中引入了 async 和 await 關鍵字,使非同步程式設計變得前所未有地簡單。
但是,ASP.NET 中的非同步功能自一開始就可以使用,只是從來沒有得到應有的重視。而且,考慮到 ASP.NET 和 IIS 處理請求的方式,非同步體現的優勢可能更明顯。透過非同步,我們很容易就可以大幅提高 ASP.NET 應用程式的擴充性。隨著新的程式結構引入,如 async 和 await 關鍵字,我們也應該學會使用非同步程式設計的強大功能。
在本篇部落格文章中,我們將討論 IIS 和 ASP.NET 處理請求的方式,然後看看 ASP.NET 中哪些地方可以使用非同步,最後再討論幾個最能體現非同步優勢的場景。
請求是如何處理的?
每個 ASP.NET 請求都要先通過 IIS,然後再由 ASP.NET 處理程序進行最終處理。 首先IIS 接收請求,初步處理後,發送給ASP.NET(必須是一個ASP.NET請求),然後由ASP.NET進行實際處理並產生回應,之後該回應會透過IIS發回給客戶。在IIS上,有一些工作進程負責從佇列中取出請求,並執行IIS 模組,然後再將該請求傳送到ASP.NET 佇列。但是,ASP.NET本身不會建立任何線程,也沒有處理請求的線程池,而是透過使用CLR 線程池,從中取得線程來處理請求。因此,IIS 模組呼叫ThreadPool.QueueUserWorkItem,將請求排入隊列,供CLR 工作執行緒處理。我們都知道,CLR執行緒池是由CLR管理,並且能夠自動調整(也就是說,它根據需要建立和銷毀進程)。這裡也要記住,建立和銷毀執行緒是項目很繁重的任務,這就是為什麼CLR執行緒池允許使用同一個執行緒處理多個任務。下面來看一個描述請求處理過程的圖示。
在上圖中可以看到,請求首先由 HTTP.sys接收,並加入到對應核心級應用程式集區佇列。然後,一個IIS工作執行緒從佇列中取出請求,處理後將其傳送到ASP.NET 佇列。注意,該請求如果不是一個ASP.NET請求,將從 IIS 自動返回。最後,從CLR線程池中分配一個線程,負責處理該請求。
ASP.NET中非同步的使用場景是?
所有請求大致可以分為兩類:
CPU Bound 類
I/O Bound 類
CPU Bound 類請求,需要CPU 時間,而且是在同一進程中執行;而I/O Bound 類請求,本身俱有阻塞性,需要依賴其他模組執行I/O 操作並回傳回應。阻塞性請求是提高應用程式可擴展性的主要障礙,而且在大多數web應用程式中,在等待 I/O 操作的過程中浪費了大量時間。 因此以下場景適合使用非同步:
I/O Bound 類別請求,包括:
資料庫存取
讀/寫檔案
Web 服務呼叫
訪問網路資源
Web 服務呼叫從多個資料來源取得資料的場景作為範例,這裡建立一個簡單的同步頁面,然後再將它轉換成非同步頁面。 本範例設定了1000ms的延遲(以模擬一些繁重的資料庫或web服務呼叫等),而且還使用WebClient下載了一個頁面,如下所示:protected void Page_Load(object sender, EventArgs e) { System.Threading.Thread.Sleep(1000); WebClient client = new WebClient(); string downloadedContent = client.DownloadString("https://msdn.microsoft.com/en-us/library/hh873175%28v=vs.110%29.aspx"); dvcontainer.InnerHtml = downloadedContent; }
<%@ Page Language="C#" AutoEventWireup="true" CodeBehind="Home.aspx.cs" Inherits="AsyncTest.Home" Async="true" AsyncTimeout="3000" %>
2.将此方法转换成异步方法。在这里把Thread.Sleep 与 client.DownloadString 转换成异步方法如下所示:
private async Task AsyncWork() { await Task.Delay(1000); WebClient client = new WebClient(); string downloadedContent = await client.DownloadStringTaskAsync("https://msdn.microsoft.com/en-us/library/hh873175%28v=vs.110%29.aspx "); dvcontainer.InnerHtml = downloadedContent; }
3.现在可以直接在 Page_Load (页面加载)上调用此方法,使其异步,如下所示:
protected async void Page_Load(object sender, EventArgs e) { await AsyncWork(); }
但是这里的 Page_Load 返回的类型是async void,这种情况无论如何都应该避免。我们知道,Page_Load 是整个页面生命周期的一部分,如果我们把它设置成异步,可能会出现一些异常情况和事件,比如生命周期已经执行完毕而页面加载仍在运行。 因此,强烈建议大家使用 RegisterAsyncTask 方法注册异步任务,这些异步任务会在生命周期的恰当时间执行,可以避免出现任何问题。
protected void Page_Load(object sender, EventArgs e) { RegisterAsyncTask(new PageAsyncTask(AsyncWork)); }
现在,页面已经转换成了异步页,它就不再是一个阻塞性请求。
笔者在 IIS8.5 上部署了同步页面和异步页面,并使用突发负载对两者进行了测试。测试结果发现,相同的机器配置,同步页面在2-3秒内只能提取1000个请求,而异步页面能够为2200多个请求提供服务。此后,开始收到超时(Timeout)或服务器不可用(Server Not Available)的错误。虽然两者的平均请求处理时间没有多大差别,但是通过异步页面,可以处理两倍以上的请求。这足以证明异步编程功能强大,所以应该充分利用它的优势。
ASP.NET中还有几个地方也可以引入异步:
编写异步模块
使用IHttpAsyncHandler 或 HttpTaskAsyncHandler 编写异步HTTP处理程序
使用web sockets 或 SignalR
结论
本篇博文中,我们讨论了异步编程,而且发现,新推出的async 和 await关键字,使异步编程变得十分简单。我们讨论的话题包括 IIS和ASP.NET如何处理请求,以及在哪些场景中异步的作用最明显。另外,我们还创建了一个简单示例,讨论了异步页面的优势。最后我们还补充了几个ASP.NET中可以使用异步的地方。

熱AI工具

Undresser.AI Undress
人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover
用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

AI Hentai Generator
免費產生 AI 無盡。

熱門文章

熱工具

記事本++7.3.1
好用且免費的程式碼編輯器

SublimeText3漢化版
中文版,非常好用

禪工作室 13.0.1
強大的PHP整合開發環境

Dreamweaver CS6
視覺化網頁開發工具

SublimeText3 Mac版
神級程式碼編輯軟體(SublimeText3)

熱門話題

C語言中通過轉義序列處理特殊字符,如:\n表示換行符。 \t表示製表符。使用轉義序列或字符常量表示特殊字符,如char c = '\n'。注意,反斜杠需要轉義兩次。不同平台和編譯器可能有不同的轉義序列,請查閱文檔。

在 C 語言中,char 類型在字符串中用於:1. 存儲單個字符;2. 使用數組表示字符串並以 null 終止符結束;3. 通過字符串操作函數進行操作;4. 從鍵盤讀取或輸出字符串。

在 C 語言中,char 和 wchar_t 的主要區別在於字符編碼:char 使用 ASCII 或擴展 ASCII,wchar_t 使用 Unicode;char 佔用 1-2 個字節,wchar_t 佔用 2-4 個字節;char 適用於英語文本,wchar_t 適用於多語言文本;char 廣泛支持,wchar_t 依賴於編譯器和操作系統是否支持 Unicode;char 的字符範圍受限,wchar_t 的字符範圍更大,並使用專門的函數進行算術運算。

C 語言中符號的使用方法涵蓋算術、賦值、條件、邏輯、位運算符等。算術運算符用於基本數學運算,賦值運算符用於賦值和加減乘除賦值,條件運算符用於根據條件執行不同操作,邏輯運算符用於邏輯操作,位運算符用於位級操作,特殊常量用於表示空指針、文件結束標記和非數字值。

多線程和異步的區別在於,多線程同時執行多個線程,而異步在不阻塞當前線程的情況下執行操作。多線程用於計算密集型任務,而異步用於用戶交互操作。多線程的優勢是提高計算性能,異步的優勢是不阻塞 UI 線程。選擇多線程還是異步取決於任務性質:計算密集型任務使用多線程,與外部資源交互且需要保持 UI 響應的任務使用異步。

在 C 語言中,char 類型轉換可以通過:強制類型轉換:使用強制類型轉換符將一種類型的數據直接轉換為另一種類型。自動類型轉換:當一種類型的數據可以容納另一種類型的值時,編譯器自動進行轉換。

char 數組在 C 語言中存儲字符序列,聲明為 char array_name[size]。訪問元素通過下標運算符,元素以空終止符 '\0' 結尾,用於表示字符串終點。 C 語言提供多種字符串操作函數,如 strlen()、strcpy()、strcat() 和 strcmp()。

C語言中沒有內置求和函數,需自行編寫。可通過遍歷數組並累加元素實現求和:循環版本:使用for循環和數組長度計算求和。指針版本:使用指針指向數組元素,通過自增指針遍歷高效求和。動態分配數組版本:動態分配數組並自行管理內存,確保釋放已分配內存以防止內存洩漏。
