Asp.Net Core 中間件應用實踐中你不知道的那些事

来源:https://www.cnblogs.com/jlion/archive/2020/03/07/12430522.html
-Advertisement-
Play Games

中間件和Filter 有哪些區別,分別的作用又是什麼,使用Endpoint 終結者路由的應用場景有哪些,怎麼使用Endpoint 終結者路由進行中間件的開發;我們知道,任何的一個web框架都是把http請求封裝成一個管道,每一次的請求都是經過管道的一系列操作,最終到達我們寫的代碼中。那麼中間件就是在... ...


一、概述

這篇文章主要分享Endpoint 終結點路由的中間件的應用場景及實踐案例,不講述其工作原理,如果需要瞭解工作原理的同學,
可以點擊查看以下兩篇解讀文章:

1.1 中間件(Middleware)的作用

我們知道,任何的一個web框架都是把http請求封裝成一個管道,每一次的請求都是經過管道的一系列操作,最終到達我們寫的代碼中。那麼中間件就是在應用程式管道中的一個組件,用來攔截請求過程進行一些其他處理和響應。中間件可以有很多個,每一個中間件都可以對管道中的請求進行攔截,它可以決定是否將請求轉移給下一個中間件。

asp.net core 提供了IApplicationBuilder介面來讓把中間件註冊到asp.net的管道請求當中去,中間件是一個典型的AOP應用。 下麵是一個微軟官方的一個中間件管道請求圖:

1.2 中間件和過濾器的區別

Filter是延續ASP.NET MVC的產物,同樣保留了五種的Filter,分別是Authorization Filter、Resource Filter、Action Filter、Exception Filter及Result Filter。
具體可以查看我上次分享的一篇Asp.Net Core Filter 深入淺出的那些事-AOP 的文章.

根據描述,可以看出中間件和過濾器的功能類似,那麼他們有什麼區別?為什麼又要搞一個中間件呢?
其實,過濾器和中間件他們的關註點是不一樣的,也就是說職責不一樣,乾的事情就不一樣。

同作為兩個AOP利器,Filter(過濾器)更貼合業務,它關註於應用程式本身,比如你看ActionFilterResultFilter,它都直接和你的ActionActionResult交互了,是不是離你很近的感覺,那我有一些比如對我的輸出結果進行格式化,對我的請求的ViewModel進行數據驗證啦,肯定就是用Filter無疑了。它是MVC的一部分,它可以攔截到你Action上下文的一些信息,而中間件是沒有這個能力的。

可以看到,每一個中間件都都可以在請求之前和之後進行操作。請求處理完成之後傳遞給下一個請求

1.3 中間件的使用場景

那麼,何時使用中間件呢?我的理解是在我們的應用程式當中和業務關係不大的一些需要在管道中做的事情可以使用,比如身份驗證,Session存儲,日誌記錄等。其實我們的 Asp.net core項目中本身已經包含了很多個中間件。比如 身份認證中間件 UseAuthorization()等系列.

二、中間件實戰

需求場景:通過後端記錄每一次的訪問請求日誌,同時需要根據需要排除一些Controller 或者Action 不記錄請求的日誌信息。

思考:經過分析我需要創建一個全局的中間件進行攔截路由,並且寫入日誌;同時需要添加一個特性Attribute 進行標註那些Controller或者Action 不需要進行日誌記錄。

我們來創建LogsMiddleware 中間件代碼,代碼如下:

public class LogsMiddleware
{
        private readonly RequestDelegate _next;

        public LogsMiddleware(RequestDelegate next)
        {
            this._next = next;
        }

        public async Task Invoke(HttpContext context)
        {
            var endpoint = context.Features.Get<IEndpointFeature>()?.Endpoint;
            if (endpoint == null)
            {
                await _next(context);
                return;
            }

            using (var scope = context.RequestServices.CreateScope())
            {
                var _logger = scope.ServiceProvider.GetService<ILogger<LogsMiddleware>>();

                var attruibutes = endpoint.Metadata.OfType<NoLogsAttriteFilter>();
                if (attruibutes.Count()==0)
                {
                    _logger.LogInformation($" url:{context.Request.Path}, 訪問時間:{DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss")}");
                }

                //記錄 排除的特殊Message 信息
                foreach (var attribute in attruibutes)
                {
                    _logger.LogInformation(attribute.Message);
                }
            }
            await _next(context);
        }
}

NoLogsAttriteFilter 過濾器代碼如下:

public class NoLogsAttriteFilter : Attribute
{
        /// <summary>
        /// 這裡加這個主要是把獲取到的信息在中間件中列印出來,區分中間件的攔截用處
        /// </summary>
        public string Message = "";

        public NoLogsAttriteFilter(string message)
        {
            Message = message;
        }
}

Startup 中的代碼如下:

public void Configure(IApplicationBuilder app, IWebHostEnvironment env)
{
            if (env.IsDevelopment())
            {
                app.UseDeveloperExceptionPage();
            }
            else
            {
                app.UseExceptionHandler("/Home/Error");
                // The default HSTS value is 30 days. You may want to change this for production scenarios, see https://aka.ms/aspnetcore-hsts.
                app.UseHsts();
            }

            app.UseRouting();
            app.UseAuthorization();
            app.UseMiddleware<LogsMiddleware>();//添加日誌記錄中間件
            app.UseEndpoints(endpoints =>
            {
                endpoints.MapControllerRoute(
                    name: "default",
                    pattern: "{controller=Home}/{action=Index}/{id?}");
            });
}

HomeController 控制器中的兩個Action 代碼如下::

// 訪問該路由會記錄訪問日誌
public IActionResult Index()
{
        return View();
}

//訪問該路由不會記錄訪問日誌
[NoLogsAttriteFilter("Manage 不需要記錄訪問日誌")]
public IActionResult Manage()
{
       return View();
}

這樣就自定義日誌中間件就已經完成了我上面的需求,不依賴於任何業務獨立存在於系統中;從代碼中我們可以看到中間件通過context.Features.Get<IEndpointFeature>()?.Endpoint; 方法獲得終結點路由方式進行匹配,並且可以通過endpoint.Metadata.OfType<NoLogsAttriteFilter>() 方式獲得Action 中的特性信息數據,並通過該攔截進行我的需求
自定義中間件教程文章請點擊自定義中間件官方教程 一文。

現在我們再來印證下我上一篇關於 Asp.Net Core EndPoint 終結點路由工作原理解讀 一文 中提及到UseRouting() 中間件是遍歷所有的Endpoint 終結點路由以匹配當前請求的 Endpoint 終結點路由一說,我把註冊LogsMiddleware中間件和UseRouting() 路由中間件代碼順序調整一下,代碼如下:

public void Configure(IApplicationBuilder app, IWebHostEnvironment env)
{
            if (env.IsDevelopment())
            {
                app.UseDeveloperExceptionPage();
            }
            else
            {
                app.UseExceptionHandler("/Home/Error");
                // The default HSTS value is 30 days. You may want to change this for production scenarios, see https://aka.ms/aspnetcore-hsts.
                app.UseHsts();
            }

            // 中間件註冊放到了UseRouting() 之前
            app.UseMiddleware<LogsMiddleware>();//添加日誌記錄中間件
            
            app.UseRouting();
            app.UseAuthorization();
            app.UseEndpoints(endpoints =>
            {
                endpoints.MapControllerRoute(
                    name: "default",
                    pattern: "{controller=Home}/{action=Index}/{id?}");
            });
}

再來看看運行調試的結果如圖:
運行調試結果

從調試的結果圖中可以看出 endpoint 變數是 null;所有需要使用到Endpoint 終結點路由必須註冊在UseRouting() 中間件之後。

三、官方常用中間件

  1. 異常/錯誤處理
    當應用在開發環境中運行時:
    開發人員異常頁中間件 (UseDeveloperExceptionPage) 報告應用運行時錯誤。
    資料庫錯誤頁中間件報告資料庫運行時錯誤。
    當應用在生產環境中運行時:
    異常處理程式中間件 (UseExceptionHandler) 捕獲以下中間件中引發的異常。
    HTTP 嚴格傳輸安全協議 (HSTS) 中間件 (UseHsts) 添加 Strict-Transport-Security 標頭。
  2. HTTPS 重定向中間件 (UseHttpsRedirection) 將 HTTP 請求重定向到 HTTPS。
  3. 靜態文件中間件 (UseStaticFiles) 返回靜態文件,並簡化進一步請求處理。
  4. Cookie 策略中間件 (UseCookiePolicy) 使應用符合歐盟一般數據保護條例 (GDPR) 規定。
  5. 用於路由請求的路由中間件 (UseRouting)。
  6. 身份驗證中間件 (UseAuthentication) 嘗試對用戶進行身份驗證,然後才會允許用戶訪問安全資源。
  7. 用於授權用戶訪問安全資源的授權中間件 (UseAuthorization)。
  8. 會話中間件 (UseSession) 建立和維護會話狀態。 如果應用使用會話狀態,請在 Cookie 策略中間件之後和 MVC 中間件之前調用會話中間件。
  9. 用於將 Razor Pages 終結點添加到請求管道的終結點路由中間件(帶有 MapRazorPages 的 UseEndpoints)。

如果您覺的不錯,請微信掃碼關註 【dotNET博士】公眾號,後續給您帶來更精彩的分享

以上如果有錯誤的地方,請大家積極糾正,謝謝大家的支持!!


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

-Advertisement-
Play Games
更多相關文章
  • 做Android手機系統或App測試的過程中, 如果碰到了bug,開發一般會需要測試人員提供當時的bug截圖, 如何用Python 批處理腳本, 快速實現截圖呢? 準備階段 1. adb shell screencap p /sdcard/a.png 命令, 可以實現手機截圖並保存到/sdcard/ ...
  • 基於 Redis 實現 CAS 操作 Intro 在 .NET 里併發情況下我們可以使用 來實現 CAS (Compare And Swap) 操作,在分散式的情景下很多時候我們都會使用 Redis ,最近在改之前做的一個微信小游戲項目,之前是單機運行的,有些數據存儲是基於記憶體的,直接基於對象操作的 ...
  • Main函數代碼 using System; namespace ConsoleApp4 { class Program { public static void Main(string[] args) { Console.WriteLine(args[0]); Console.WriteLine( ...
  • VS2019 for MAC已經發佈很長時間了,本以為項目移過去很麻煩,一直沒有動作,最近呆家裡快發黴了,決定研究研究,沒想到一句代碼都不需要動,直接完功,這下可以生產了。同學們可以放心整了。 ...
  • HSQL 是一種輕量級的基於 .NET Core 的資料庫對象關係映射「ORM」框架 HSQL 是一種可以使用非常簡單且高效的方式進行資料庫操作的一種框架,通過簡單的語法,使資料庫操作不再成為難事。目前支持的資料庫有 MySql、SQLServer。 安裝方法 Install-Package HSQ ...
  • 問題: 其實也不是問題了 算是優化吧 當做net項目時 不是前後臺分離時 需要寫很多的前端頁面 這時我們就會用到很多的ui插件js,css文件 (這裡指的第三方的ui插件不是自己寫的js,css) 比如bootstrap 當我們新建一個mvc項目時 會自動下載bootstrap 但是用的多了之後 j ...
  • 前言 上一篇博客上已經實現了使用EventBus對具體事件行為的分發處理,某種程度上也算是基於事件驅動思想編程了。但是如上篇博客結尾處一樣,我們源碼的執行效率依然達不到心裡預期。在下單流程里我們明顯可以將部分行為進行非同步處理,提升下單操作的執行效率。 Redis基礎命令 Redis有兩種方式可支持我 ...
  • 在上一篇文章Abp(net core)+easyui+efcore實現倉儲管理系統——入庫管理之二(三十八) 中我們創建了入庫單的一些有關DTO類與分頁類。由於入庫單我使用了到了資料庫的存儲過程,那麼本篇文章中我們來學習一下如何在ABP中調用存儲過程。 我們都知道,倉儲管理系統中的單號最基本... ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...