成人在线亚洲_国产日韩视频一区二区三区_久久久国产精品_99国内精品久久久久久久

您的位置:首頁技術(shù)文章
文章詳情頁

手把手帶你定制.NET 6.0的Middleware中間件

瀏覽:289日期:2022-06-09 09:01:55
目錄
  • 前言
  • 技術(shù)準(zhǔn)備
  • 中間件簡介
  • 編寫自定義中間件
  • 中間件的潛力
    • 使用/map分支管道
  • 使用MapWhen分支管道
    • 使用中間件構(gòu)造條件
  • 在ASP.NET Core 3.0及更高版本中使用中間件
    • 重寫終止中間件
  • 總結(jié)

    前言

    在本文中,我們將學(xué)習(xí)中間件,以及如何使用它進(jìn)一步定制應(yīng)用程序。我們將快速學(xué)習(xí)中間件的基礎(chǔ)知識,然后探討如何使用它做的一些特殊事情。
    本文涵蓋的主題包括:

    • 中間件簡介
    • 編寫自定義中間件
    • 中間件的潛力
    • 如何使用中間件

    本章所處的位置,如下圖所示:

    技術(shù)準(zhǔn)備

    我們使用控制臺、shell或Bash終端先創(chuàng)建一個(gè)ASP.NET Core MVC應(yīng)用程序,然后切換到工作目錄:

    dotnet new web -n MiddlewaresDemo -o MiddlewaresDemo

    然后用VS打開項(xiàng)目:

    cd MiddlewaresDemo code .

    注意在.NET 6.0中,web項(xiàng)目模板發(fā)生了變化。Microsoft引入了minimal API,項(xiàng)目模板默認(rèn)使用minimal API。

    中間件簡介

    大多數(shù)人可能已經(jīng)知道中間件是什么,但有些人可能不知道,即使你已經(jīng)在使用ASP.NET Core有一段時(shí)間了。我們一般不需要詳細(xì)了解中間件實(shí)例,因?yàn)樗鼈兇蠖嚯[藏在擴(kuò)展方法后面,例如UseMvc()、UseAuthentication()、UseDeveloperExceptionPage()等。每次在Configure方法中,我們默認(rèn)將隱式地使用至少一個(gè)或更多個(gè)中間件組件。

    中間件組件是處理請求管道的一段代碼。我們可以將請求流程想象成一串管道,每次請求調(diào)用,都會返回一個(gè)響應(yīng)。中間件負(fù)責(zé)創(chuàng)建回聲——它操縱請求上下文,加工處理、疊加邏輯、豐富信息。

    中間件組件按配置順序執(zhí)行。配置的第一個(gè)中間件組件是第一個(gè)執(zhí)行的組件。我們可以把中間件看成回旋鏢,出去的時(shí)候第一個(gè)執(zhí)行,回來的時(shí)候最后一個(gè)執(zhí)行。

    在ASP.NET Core web應(yīng)用程序,如果客戶端請求的是圖像或任何其他靜態(tài)文件,StaticFileMiddleware將負(fù)責(zé)查找該資源,如果找到該資源,則返回該資源。如果沒有,這個(gè)中間件除了調(diào)用下一個(gè)之外什么都不做。

    MvcMiddleware組件檢查請求的資源,將其映射到已配置的路由,執(zhí)行控制器,創(chuàng)建視圖,并返回HTML或Web API結(jié)果。如果MvcMiddleware沒有找到匹配的控制器,它無論如何都會返回一個(gè)結(jié)果——通常是一個(gè)404狀態(tài)的結(jié)果,這就是為什么MvcMiddleware是最后配置的中間件。

    異常處理中間件通常是配置的第一批的中間件之一,不是因?yàn)樗堑谝粋€(gè)執(zhí)行的,而是因?yàn)樗亲詈笠粋€(gè)執(zhí)行的。異常處理驗(yàn)證結(jié)果,并以客戶端友好的方式在瀏覽器中顯示可能的異常。以下過程描述了運(yùn)行時(shí)發(fā)生的500錯(cuò)誤狀態(tài):

    var builder = WebApplication.CreateBuilder(args); var app = builder.Build(); app.MapGet("/", () => "Hello World!"); app.Run();

    在ASP.NET Core 6.0,Microsoft引入了minimal API,它簡化了應(yīng)用配置,并隱藏了許多默認(rèn)配置,比如隱式的using聲明,因此,在頭部我們看不到任何using語句,以上就是我們看到的ASP.NET Core 6.0中的Program.cs文件內(nèi)容。

    在這里,lambda中間件綁定到默認(rèn)路由,只有一句簡單的“Hello World!”響應(yīng)流。這個(gè)特殊的中間件會終止管道并返回響應(yīng)內(nèi)容。因此,它是最后一個(gè)運(yùn)行的中間件。

    下面我們把a(bǔ)pp.MapGet()做個(gè)替換,如下所示:

    app.Use(async (context, next) =>{         await context.Response.WriteAsync("===");         await next();         await context.Response.WriteAsync("==="); }); app.Use(async (context, next) => {     await context.Response.WriteAsync(">>>>>> ");         await next();         await context.Response.WriteAsync(" <<<<<<");}); app.Run(async context => {     await context.Response.WriteAsync("Hello World!"); });

    這里調(diào)用兩個(gè)app.Use()方法,并且創(chuàng)建了兩個(gè)lambda中間件,除了做簡單的處理外,中間件還調(diào)用了它們的后繼組件,每個(gè)中間件的調(diào)用鏈很明確很清晰。在調(diào)用下一個(gè)中間件之前,處理實(shí)際的請求,在調(diào)用下個(gè)中間件之后,處理響應(yīng)。以上就是管道的工作機(jī)制。

    如果現(xiàn)在運(yùn)行程序(使用dotnet run)并在瀏覽器中打開URL,我們應(yīng)該會看到這樣的純文本結(jié)果

    ===>>>>>> Hello World! <<<<<<===

    不知道您理解了沒?如果理解了,我們往下學(xué)習(xí),看看如何使用這個(gè)概念向請求管道添加一些附加功能。

    編寫自定義中間件

    中間件可以說是ASP.NET Core的基座,在請求期間執(zhí)行的所有邏輯都基于此機(jī)制。因此,我們可以使用它向web添加自定義功能。在下面案例,我們希望找出通過請求管道的每個(gè)請求的執(zhí)行時(shí)間:

    我們可以在調(diào)用下一個(gè)中間件之前創(chuàng)建并啟動(dòng)秒表,然后在調(diào)用下個(gè)中間件之后停止測量執(zhí)行時(shí)間,如下所示:

    app.Use(async (context, next) => {         var s = new Stopwatch();         s.Start();         //其他操作     await next();         s.Stop();     //結(jié)束度量         var result = s.ElapsedMilliseconds;         //統(tǒng)計(jì)耗時(shí)         await context.Response.WriteAsync($"耗時(shí):{result} 秒。"); });

    記得為System.Diagnostics添加using語句。

    之后,我們將經(jīng)過的毫秒返回到響應(yīng)流。

    如果您編寫的中間件組件很多,Program.cs將變得非常混亂。所以大多數(shù)中間件組件將被編寫為獨(dú)立的類,如下所示:

    using System.Diagnostics; public class StopwatchMiddleware { ???    private readonly RequestDelegate _next; ????    public StopwatchMiddleware(RequestDelegate next) ?    { ?_next = next; ?    } ????    public async Task Invoke(HttpContext context) { ????????var s = new Stopwatch(); ????????s.Start(); ????????//其他操作  ????????await _next(context); ????????s.Stop(); //結(jié)束度量 ????????var result = s.ElapsedMilliseconds; ????????//統(tǒng)計(jì)耗時(shí)     await context.Response.WriteAsync($"耗時(shí):{result} 秒。");  ??    }  }

    在Invoke方法中的,我們獲得構(gòu)造函數(shù)和當(dāng)前上下文獲得要執(zhí)行的下一個(gè)中間件組件。

    注意:

    中間件在應(yīng)用程序啟動(dòng)時(shí)初始化,構(gòu)造函數(shù)在應(yīng)用程序生命周期內(nèi)僅運(yùn)行一次。另一方面,每個(gè)請求調(diào)用一次Invoke方法。

    要使用此中間件,您可以使用一個(gè)通用的UseMiddleware方法:

    app.UseMiddleware<StopwatchMiddleware>();

    然而,更優(yōu)雅的方法是創(chuàng)建一個(gè)封裝此調(diào)用的擴(kuò)展方法:

    public static class StopwatchMiddlewareExtension {         public static IApplicationBuilder  UseStopwatch(this IApplicationBuilder app)         { app.UseMiddleware<StopwatchMiddleware>(); return app;         } }

    然后就可以這樣使用:

    app.UseStopwatch();

    這樣,您可以通過請求管道向ASP.NET Core應(yīng)用程序提供其他功能。中間件中提供了整個(gè)HttpContext。這樣,您可以使用中間件操縱請求和響應(yīng)。

    例如,AuthenticationMiddleware嘗試從請求中收集用戶信息。如果找不到任何信息,它將通過向客戶端發(fā)送特定的響應(yīng)來請求信息。如果它找到,它會將其添加到請求上下文中,并以這種方式將其提供給整個(gè)應(yīng)用程序。

    中間件的潛力

    使用中間件還可以做許多其他事情。例如,可以將請求管道拆分為兩個(gè)或多個(gè)管道,我們將在這里討論如何做到這一點(diǎn)。

    使用/map分支管道

    下一段代碼顯示了如何基于特定路徑創(chuàng)建請求管道的分支:

    app.Map("/map1", app1 => {         // 其他中間件         app1.Run(async context =>     {     await context.Response.WriteAsync("Map Test 1");         }); }); app.Map("/map2", app2 => {          // 其他中間件         app2.Run(async context => { await context.Response.WriteAsync("Map Test 2");         }); }); // 其他中間件

    /map1路徑是一個(gè)特定的分支,它在內(nèi)部繼續(xù)請求管道,/map2與此相同。這兩個(gè)map都有自己內(nèi)部的中間件配置。所有其他未指定的路徑都遵循該主分支。

    使用MapWhen分支管道

    還有一個(gè)MapWhen方法可以根據(jù)條件分支管道,而不是根據(jù)路徑分支:

    public void Configure(IApplicationBuilder app) {         app.MapWhen(context =>context.Request.Query.ContainsKey("分支"), app1 => {         // 其他中間件       app1.Run(async context =>  {  await context.Response.WriteAsync( "MapBranch Test");     });         });         //其他中間件         app.Run(async context =>    { await context.Response.WriteAsync("Hello non-Map.");         });}

    使用中間件構(gòu)造條件

    我們一般可以根據(jù)配置值創(chuàng)建條件,或者根據(jù)請求上下文的屬性創(chuàng)建條件。在前面的示例中,我們使用了查詢字符串屬性作為條件。當(dāng)然,你也可以使用HTTP標(biāo)頭、表單屬性或請求上下文的任何其他屬性。

    如果需要,還可以嵌套map以創(chuàng)建子分支和孫分支。

    我們再看下健康檢查中間件,ASP.NET Core HealthCheck API的工作原理如下:

    首先,它使用MapWhen指定要使用的端口,然后,它使用Map設(shè)置HealthCheck API路徑(如果未指定端口則使用Map)。最后,使用了HealthCheckMiddleware。我們看下面的代碼示例:

    private static void UseHealthChecksCore(IApplicationBuilder app, PathString path, int? port, object[] args) {         if (port == null)        { app.Map(path, b => b.UseMiddleware<HealthCheckMiddleware>(args));     }         else  {app.MapWhen(c => c.Connection.LocalPort == port,    b0 => b0.Map(path, b1 =>b1.UseMiddleware<HealthCheckMiddleware>(args)));         }; }

    這里,我們可以使用Map或MapWhen分別基于特定路徑或特定條件提供特殊的API或資源。

    接下來,讓我們看看如何在更新版本的ASP.NET Core中使用終止中間件組件。

    在ASP.NET Core 3.0及更高版本中使用中間件

    ASP.NET Core 3.0及更高版本,有兩種新的中間件,它們被稱為UseRoutingUseEndpoints

    public void Configure(IApplicationBuilder app, IWebHostEnvironment env) {         if (env.IsDevelopment())         {  app.UseDeveloperExceptionPage();         }         app.UseRouting();         app.UseEndpoints(endpoints =>  { endpoints.MapGet("/", async context => {     await context.Response.WriteAsync("Hello  World!"); });         }); }

    第一個(gè)是使用路由的中間件UseRouting,另一個(gè)是訪問地址的UseEndpoints

    這是新的端點(diǎn)路由。以前,路由是MVC的一部分,它只適用于MVC、Web API和基于MVC的框架。然而在ASP.NET Core 3.0及更高版本,路由不再是MVC框架中的一部分。現(xiàn)在,MVC和其他框架都可以被映射到特定的路由或端點(diǎn)。

    在前面的代碼段中,GET請求被映射到頁面根URL。在下一個(gè)代碼片段中,MVC被映射到路由模式,RazorPages被映射到基于RazorPage的特定文件結(jié)構(gòu)的路由:

    app.UseEndpoints(endpoints => {         endpoints.MapControllerRoute(name: "default", pattern: "{controller=Home}/{action=Index}/{id?}");     endpoints.MapRazorPages(); });

    現(xiàn)在已經(jīng)沒有UseMvc方法了,即使它仍然存在并在IApplicationBuilder對象級別上工作,以防止現(xiàn)有代碼中斷。現(xiàn)在,激活A(yù)SP.NET Core功能的方法更為精細(xì)。

    • Areas for MVC and web API: endpoints.MapAreaControllerRoute(...);
    • MVC and web API: endpoints.MapControllerRoute(...);
    • Blazor server-side: endpoints.MapBlazorHub(...);
    • SignalR: endpoints.MapHub(...);
    • Razor Pages: endpoints.MapRazorPages(...);
    • Health checks: endpoints.MapHealthChecks(...);

    這些是ASP最常用的新Map方法。

    還有很多方法可以定義回退地址,比如將路由和HTTP方法映射到代理,以及中間件組件。

    你可以創(chuàng)建適用于所有請求的中間件,例如StopWatchMiddleware,你也可以編寫中間件以在特定路徑或路由上工作,例如創(chuàng)建一個(gè)Map方法,以將其映射到該路由。

    注意事項(xiàng)
    不再建議在中間件內(nèi)部處理路由。相反,您應(yīng)該使用新的地址路由。使用這種方法,中間件更加通用,它可以通過單一的配置就可以在多個(gè)路由上工作。

    重寫終止中間件

    接下來,我們創(chuàng)建小型虛擬中間件,將應(yīng)用程序狀態(tài)寫入特定路由。在此示例中,沒有自定義路由處理:

    namespace MiddlewaresSample; public class AppStatusMiddleware {         private readonly RequestDelegate _next;         private readonly string _status;    public AppStatusMiddleware(RequestDelegate next, string status)         {_next = next; _status = status;        }         public async Task Invoke(HttpContext context)  { await context.Response.WriteAsync($"Hello {_status}!");         } }

    我們需要做的是在IEndpointRouteBuilder對象上編寫一個(gè)擴(kuò)展方法。此方法將路由模式作為可選參數(shù),并返回IEndpointConventionBuilder對象以啟用跨域資源共享(CORS)、身份驗(yàn)證或路由的其他條件。

    現(xiàn)在,我們應(yīng)該添加一個(gè)擴(kuò)展方法,以便更容易地使用中間件:

    public static class MapAppStatusMiddlewareExtension {         public static IEndpointConventionBuilder MapAppStatus(this IEndpointRouteBuilder routes, string pattern = "/", string name = "World")      { var pipeline = routes.CreateApplicationBuilder().UseMiddleware<AppStatusMiddleware>(name).Build(); return routes.Map(pattern, pipeline).WithDisplayName("AppStatusMiddleware");         } }

    完成后,我們可以使用MapAppStatus方法將其映射到特定路線:

    app.UseRouting(); app.UseEndpoints(endpoints => {         endpoints.MapGet("/", () => "Hello World!");         endpoints.MapAppStatus("/status", "Status"); });

    現(xiàn)在,我們可以通過輸入以下地址在瀏覽器中調(diào)用路由: http://localhost:5000/status

    總結(jié)

    大多數(shù)ASP.NET Core功能基于中間件,在本章中,我們學(xué)習(xí)了中間件的工作原理以及如何創(chuàng)建自己的中間件組件來擴(kuò)展ASP.NET框架。我們還學(xué)習(xí)了如何使用新路由向自定義的終止中間件添加路由。

    在下一章中,我們將了解ASP.NET Core中的新端點(diǎn)路由,它允許我們以簡單靈活的方式創(chuàng)建自己的托管端點(diǎn)。

    到此這篇關(guān)于定制.NET 6.0的Middleware中間件的文章就介紹到這了,更多相關(guān)定制.NET 6.0 Middleware中間件內(nèi)容請搜索以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持!

    標(biāo)簽: ASP.NET
    成人在线亚洲_国产日韩视频一区二区三区_久久久国产精品_99国内精品久久久久久久
    午夜精品久久久久久久99水蜜桃 | 久久se精品一区精品二区| 欧美精品在线一区| 日韩一区二区三区免费看| 精品一区二区三区久久久| 色噜噜狠狠成人中文综合| 亚洲激情一二三区| 亚洲激情社区| 亚洲品质自拍视频网站| 亚洲国产精品www| 亚洲色图自拍偷拍美腿丝袜制服诱惑麻豆| 91麻豆国产福利在线观看| 亚洲色图第一区| 久久久青草青青国产亚洲免观| 色8久久人人97超碰香蕉987| 最新亚洲激情| 欧美伊人久久| 丁香六月久久综合狠狠色| 蜜臀av性久久久久蜜臀av麻豆| 亚洲天堂av一区| 国产欧美一区二区在线观看| 欧美日韩视频专区在线播放| 亚洲一级在线| 一本色道久久综合| 亚洲性图久久| 午夜久久美女| 欧美日韩亚洲一区二区三区在线观看 | 日本道色综合久久| 在线视频一区观看| 国产精品yjizz| 色综合视频一区二区三区高清| 国产精品白丝jk黑袜喷水| 日韩av在线发布| 午夜a成v人精品| 亚洲一区二区三区国产| 中文字幕亚洲欧美在线不卡| 国产精品美女一区二区三区| 欧美成人女星排名| 欧美va亚洲va国产综合| 日韩一区二区三区高清免费看看| 欧洲人成人精品| 欧美三级日韩三级国产三级| 欧美无乱码久久久免费午夜一区 | 欧美性受xxxx黑人xyx| 国产精品美女久久久浪潮软件| 亚洲精品日韩久久| 亚洲激情网址| 一区二区精品在线观看| 国产精品一区二区欧美| 校园激情久久| 欧美亚洲一区二区三区四区| 国产精品永久入口久久久| 五月天久久比比资源色| 欧美日韩中文另类| 99精品久久只有精品| 亚洲国产精品综合小说图片区| 亚洲成人在线网站| 国产在线不卡一区| 国产成人丝袜美腿| 欧美日韩亚洲一区二区三区在线| 在线电影一区| 在线视频欧美精品| 日韩一区二区精品在线观看| 国产欧美日韩卡一| 亚洲午夜国产一区99re久久| 美女网站视频久久| 99久久99久久久精品齐齐| 国产在线视频欧美一区二区三区| 99精品国产在热久久| 精品视频在线免费看| 亚洲精品一区二区三区蜜桃下载| 亚洲国产岛国毛片在线| 亚洲午夜国产一区99re久久| 开心九九激情九九欧美日韩精美视频电影 | 亚洲激情中文1区| 秋霞国产午夜精品免费视频| 成人性生交大合| 精品动漫3d一区二区三区免费| 久久国产日韩欧美| 日韩一区二区在线看片| 中文字幕亚洲视频| 日韩二区三区四区| 91在线国内视频| 极品av少妇一区二区| 国产精品高潮呻吟| 91国在线观看| 91麻豆视频网站| 久久精品av麻豆的观看方式| 亚洲国产视频在线| 99久久er热在这里只有精品66| 亚洲一区二区高清视频| 日韩三级在线观看| 亚洲一区二区三区四区在线| 成人黄色小视频| 亚洲在线成人| 久久精品夜色噜噜亚洲aⅴ| 亚洲五码中文字幕| 99久久99久久精品免费观看 | 91精品91久久久中77777| 99久久免费视频.com| 亚洲国产精品久久久久秋霞影院| 欧美成人精品1314www| 亚洲专区欧美专区| 成人h动漫精品| 三级不卡在线观看| 欧美国产一区二区在线观看| 欧美日韩中文字幕精品| 影音先锋中文字幕一区| 国产成人午夜99999| 亚洲二区在线观看| 中文一区一区三区高中清不卡| 久久99精品一区二区三区三区| 国产超碰在线一区| 在线观看91精品国产入口| 亚洲男同性视频| 欧美精品二区| 久久亚洲综合色一区二区三区| 久久99久国产精品黄毛片色诱| 国产精品一区二区久激情瑜伽| 亚洲小说欧美激情另类| 国产午夜一区二区三区| 欧美乱妇一区二区三区不卡视频 | 欧美一区二区视频观看视频 | av成人免费观看| 亚洲男人的天堂在线aⅴ视频| 51精品国自产在线| 老司机一区二区三区| av日韩在线网站| 另类小说欧美激情| 亚洲在线中文字幕| 国产精品久久二区二区| 欧美成人性战久久| 91国偷自产一区二区开放时间| 亚洲欧洲在线一区| 欧美区亚洲区| av在线播放不卡| 国产乱子轮精品视频| 日本成人在线一区| 亚洲国产一区二区在线播放| 国产精品久久久久桃色tv| 精品免费99久久| 在线成人av网站| 国产日产欧美一区二区三区| 国产成人精品综合在线观看| 新67194成人永久网站| 豆国产96在线|亚洲| 狠狠色狠狠色综合| 日韩电影一二三区| 亚洲网友自拍偷拍| 亚洲欧美激情插| 99成人在线| 国产精品久久久久久户外露出| 伊人成人在线视频| 亚洲最色的网站| 性8sex亚洲区入口| 精品一区二区在线播放| 欧美一区二视频| 国产精品一级| 国产在线欧美日韩| 欧美日韩国产高清视频| 91视视频在线观看入口直接观看www| 国产尤物一区二区在线| 激情五月婷婷综合| 激情欧美日韩一区二区| 美国精品在线观看| 秋霞成人午夜伦在线观看| 日韩影院免费视频| 日韩黄色在线观看| 免费欧美日韩国产三级电影| 欧美色国产精品| 牛人盗摄一区二区三区视频| 亚洲综合无码一区二区| 欧美日韩一级黄| 欧美日韩国产不卡在线看| 亚洲资源在线观看| 欧美一区二区久久久| 国内综合精品午夜久久资源| 丝袜美腿亚洲一区| 国产福利一区二区三区视频在线 | 青青草国产精品亚洲专区无| 欧美人妖巨大在线| 欧美日韩亚洲一区| 日韩黄色免费电影| 日韩精品自拍偷拍| 一区二区三区国产盗摄| 国产高清视频一区| 夜夜操天天操亚洲| 26uuu另类欧美| 91久久精品午夜一区二区| 欧美1区3d| 久久国产视频网| 亚洲人一二三区| 伊人精品在线| av电影在线不卡| 欧美99在线视频观看| 国内精品久久久久久久影视蜜臀| 亚洲美女91| 日本高清无吗v一区| 欧美精品久久天天躁| 欧美精品一区二区三区蜜桃视频|