.NET Core中間件的註冊和管道的構建(1)---- 註冊和構建原理

来源:http://www.cnblogs.com/durow/archive/2016/08/04/5736385.html
-Advertisement-
Play Games

.NET Core中間件的註冊和管道的構建(1) 註冊和構建原理 0x00 問題的產生 管道是.NET Core中非常關鍵的一個概念,很多重要的組件都以中間件的形式存在,包括許可權管理、會話管理、路由等。所以搞明白中間件是如何註冊並最終構建成管道的很重要。園子里很多先驅早已經開始了這方面的研究學習,也 ...


.NET Core中間件的註冊和管道的構建(1)---- 註冊和構建原理

0x00 問題的產生

管道是.NET Core中非常關鍵的一個概念,很多重要的組件都以中間件的形式存在,包括許可權管理、會話管理、路由等。所以搞明白中間件是如何註冊並最終構建成管道的很重要。園子里很多先驅早已經開始了這方面的研究學習,也寫了很多文章,不過我看了後有些地方還不是特別明白。畢竟每個人都是不同的,有些內容作者覺得是常識不需要多寫的地方對我來說可能就是個盲區。幸好.NET Core整個項目都是開源的,找到源碼看了下解決了我心中的困惑。同時寫篇博客記錄一下,也算一個補充,如果大家看了能有所收穫那就更好了。本來是想一片文章寫完的,後來發現太長了,所以分了兩篇。這是第一篇,主要說一下中間件的註冊和管道的構建原理,後面一篇寫一下註冊中間件類的原理和寫中間件類需要註意的約定和特性。

0x01 中間件的註冊

管道的構建主要包含中間件註冊和把註冊的中間件構建成管道。先來說中間件的註冊。

什麼是中間件。說簡單一點中間件就是一個方法,傳入一個HttpContext類型參數,返回Task。參數HttpContext中包含了HTTP請求和響應等相關信息,中間件可以讀取/修改其中的部分內容,並決定是否讓下一個中間件繼續處理這個HttpContext。這個方法包裝為一個委托RequestDelegate。

 

為了讓中間件有權決定是否讓下一個中間件繼續處理HttpContext,當前中間件需要下一個中間件的引用。所以在註冊中間件的時候需要註冊為Func<RequestDelegate,RequestDelegate>的形式,其中傳入的參數指的是下一個中間件,返回的是我們註冊的中間件,傳入的參數在ApplicationBuilder在執行Build()方法構建管道時傳入(後面會看到)。所以為了方便理解,不那麼嚴謹的來看,中間件以Func<NextMiddleware,ThisMiddleware>的形式註冊並存儲在一個列表中。

 

 可以通過ApplicationBuilder的Use方法註冊中間件

0x02 管道的構建

管道的構建就是把Func<RequestDelegate,RequestDelegate>列表串在一起。用第一個Func的返回值作為第二個Func的參數,第二個Func的返回值作為第三個Func的參數,依次類推。最終返回最後一個Func返回的RequestDelegate(中間件)。管道工作時最後一個RequestDelegate可以決定是否調用倒數第二個RequestDelegate,倒數第二個RequestDelegate可以決定是否調用倒數第三個RequestDelegate,依次類推直到調用第一個RequestDelegate。這中間有兩個問題:

第一個是這樣構建管道,中間件的順序和註冊的時候是相反的,所以在構件時首先把列表_components.Reverse()以保證正確的中間件順序。

第二個問題是構建第一個中間件時沒有RequestDelegate可以傳入,所以需要構建一個把狀態碼設置為404的中間件作為最開始的RequestDelegate傳入。當然在構建完成後這個中間件是存在於管道最末端的。這樣管道構建就算完成了。Build()代碼如下:

0x03 測試

構建完成後的管道和中間件如下圖:

這是微軟官方的圖,很多文章中也引用過。從這張圖中可以看出來中間件通過調用next()啟動下一個中間件。如果中間件不調用next()那麼它之後的所有中間件就都不會調用了。除此之外還有一個細節需要註意,就是next()的調用並不是必須要放到最後的。也就是說可以先調用後面的中間件,等後面的中間件調用完成後在執行一些操作(圖中的more logic)。

下麵來分別進行測試,正常建立一個.NET Core MVC Web項目。

測試1:註釋掉Configure()中的所有內容,然後依次註冊中間件:

運行後結果為:

這個測試印證了之前代碼中看到的中間件的註冊順序就是調用順序。

 


 

測試2:註釋掉Middleware1的next調用,其它保持不變。這樣Middleware1就不會調用Middleware2,Middleware2以及之後的所有中間件都無法調用。

運行結果變為:

這個測試說明Middleware1不調用next()的話後面的Middleware2和Middleware3都沒有被調用。

 


 

測試3:把Configure()方法修改如下:

我們在最開始註冊一個中間件記錄當前時間,然後調用後面所有中間件,最後返回時計算後面所有中間件執行所消耗的時間。為了看上去更明顯後面又註冊了一個中間件強制睡眠100毫秒。需要註意的是強制睡眠的中間件要註冊在MVC之前,因為MVC結束後就直接返回了,不會調用後面的中間件了。

運行結果為:

這個測試說明對下一個中間件的調用不一定非要放到最後,可以先調用後面中間件,等後面所有中間件調用完成後再繼續處理。

0x04 寫在最後

這篇文章主要討論了中間件的註冊和管道構建的一些原理,實際上對於複雜一點的中間件來說,一般都有更複雜的邏輯並對其它組件依賴。下一篇將討論把中間件寫成一個類並註入依賴的方法和原理。

此外這篇文章主要是我個人的一些理解和直覺。。。好吧真的有些是直覺,能力有限,博客園大牛眾多,有錯誤的地方大家嘴下留情啊。


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

-Advertisement-
Play Games
更多相關文章
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...