亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > .NET > 正文

詳解ASP.NET Core MVC 源碼學習:Routing 路由

2024-07-10 13:32:11
字體:
來源:轉載
供稿:網友

前言

最近打算抽時間看一下 ASP.NET Core MVC 的源碼,特此把自己學習到的內容記錄下來,也算是做個筆記吧。

路由作為 MVC 的基本部分,所以在學習 MVC 的其他源碼之前還是先學習一下路由系統,ASP.NET Core 的路由系統相對于以前的 Mvc 變化很大,它重新整合了 Web Api 和 MVC。

路由源碼地址 :Routing-dev.rar

路由(Routing)功能介紹

路由是 MVC 的一個重要組成部分,它主要負責將接收到的 Http 請求映射到具體的一個路由處理程序上,在MVC 中也就是說路由到具體的某個 Controller 的 Action 上。

路由的啟動方式是在ASP.NET Core MVC 應用程序啟動的時候作為一個中間件來啟動的,詳細信息會在下一篇的文章中給出。

通俗的來說就是,路由從請求的 URL 地址中提取信息,然后根據這些信息進行匹配,從而映射到具體的處理程序上,因此路由是基于URL構建的一個中間件框架。
 路由還有一個作用是生成響應的的URL,也就是說生成一個鏈接地址可以進行重定向或者鏈接。

路由中間件主要包含以下幾個部分:

  1. URL 匹配
  2. URL 生成
  3. IRouter 接口
  4. 路由模板
  5. 模板約束

Getting Started

ASP.NET Core Routing 主要分為兩個項目,分別是 Microsoft.AspNetCore.Routing.Abstractions,Microsoft.AspNetCore.Routing。前者是一個路由提供各功能的抽象,后者是具體實現。

我們在閱讀源碼的過程中,我建議還是先大致瀏覽一下項目結構,然后找出關鍵類,再由入口程序進行閱讀。

Microsoft.AspNetCore.Routing.Abstractions

大致看完整個結構之后,我可能發現了幾個關鍵的接口,理解了這幾個接口的作用后能夠幫助我們在后續的閱讀中事半功倍。

IRouter

Microsoft.AspNetCore.Routing.Abstractions 中有一個關鍵的接口就是 IRouter:

public interface IRouter{ Task RouteAsync(RouteContext context); VirtualPathData GetVirtualPath(VirtualPathContext context);}

這個接口主要干兩件事情,第一件是根據路由上下文來進行路由處理,第二件是根據虛擬路徑上下文獲取 VirtualPathData

IRouteHandler

另外一個關鍵接口是 IRouteHandler , 根據名字可以看出主要是對路由處理程序機型抽象以及定義的一個接口。

public interface IRouteHandler{ RequestDelegate GetRequestHandler(HttpContext httpContext, RouteData routeData);}

它返回一個 RequestDelegate 的一個委托,這個委托可能大家比較熟悉了,封裝了處理Http請求的方法,位于Microsoft.AspNetCore.Http.Abstractions 中,看過我之前博客的同學應該比較了解。

這個接口中 GetRequestHandler 方法有兩個參數,第一個是 HttpContext,就不多說了,主要是來看一下第二個參數 RouteData。

RouteData,封裝了當前路由中的數據信息,它包含三個主要屬性,分別是 DataTokens, Routers, Values。

DataTokens: 是匹配的路徑中附帶的一些相關屬性的鍵值對字典。

Routers: 是一個 Ilist<IRouter> 列表,說明RouteData 中可能會包含子路由。

Values: 當前路由的路徑下包含的鍵值。

還有一個 RouteValueDictionary, 它是一個集合類,主要是用來存放路由中的一些數據信息的,沒有直接使用 IEnumerable<KeyValuePair<string, string>> 這個數據結構是應為它的內部存儲轉換比較復雜,它的構造函數接收一個 Object 的對象,它會嘗試將Object 對象轉化為自己可以識別的集合。

IRoutingFeature

我根據這個接口的命名一眼就看出來了這個接口的用途,還記得我在之前博客中講述Http管道流程中得時候提到過一個叫 工具箱 的東西么,這個 IRoutingFeature 也是其中的一個組成部分。我們看一下它的定義:

public interface IRoutingFeature{ RouteData RouteData { get; set; }}

原來他只是包裝了 RouteData,到 HttpContext 中啊。

IRouteConstraint

這個接口我在閱讀的時候看了一下注釋,原來路由中的參數參數檢查主要是靠這個接口來完成的。

我們都知道在我們寫一個 Route Url地址表達式的時候,有時候會這樣寫:Route("/Product/{ProductId:long}") , 在這個表達式中有一個 {ProductId:long} 的參數約束,那么它的主要功能實現就是靠這個接口來完成的。

/// Defines the contract that a class must implement in order to check whether a URL parameter/// value is valid for a constraint.public interface IRouteConstraint{ bool Match(  HttpContext httpContext,  IRouter route,  string routeKey,  RouteValueDictionary values,  RouteDirection routeDirection);}

Microsoft.AspNetCore.Routing

Microsoft.AspNetCore.Routing 主要是對 Abstractions 的一個主要實現,我們閱讀代碼的時候可以從它的入口開始閱讀。

RoutingServiceCollectionExtensions 是一個擴展ASP.NET Core DI 的一個擴展類,在這個方法中用來進行 ConfigService,Routing 對外暴露了一個 IRoutingBuilder 接口用來讓用戶添加自己的路由規則,我們來看一下:

public static IApplicationBuilder UseRouter(this IApplicationBuilder builder, Action<IRouteBuilder> action){ //...略  // 構造一個RouterBuilder 提供給action委托宮配置 var routeBuilder = new RouteBuilder(builder); action(routeBuilder);  //調用下面的一個擴展方法,routeBuilder.Build() 見下文 return builder.UseRouter(routeBuilder.Build());}public static IApplicationBuilder UseRouter(this IApplicationBuilder builder, IRouter router){  //...略   return builder.UseMiddleware<RouterMiddleware>(router);}

routeBuilder.Build() 構建了一個集合 RouteCollection,用來保存所有的 IRouter 處理程序信息,包括用戶配置的Router。

RouteCollection 本身也實現了 IRouter , 所以它也具有路由處理的能力。

Routing 中間件的入口是 RouterMiddleware 這個類,通過這個中間件注冊到 Http 的管道處理流程中, ASP.NET Core MVC 會把它默認的作為其配置項的一部分,當然你也可以把Routing單獨拿出來使用。

我們來看一下 Invoke 方法里面做了什么,它位于RouterMiddleware.cs 文件中。

public async Task Invoke(HttpContext httpContext){  var context = new RouteContext(httpContext);  context.RouteData.Routers.Add(_router);  await _router.RouteAsync(context);  if (context.Handler == null)  {    _logger.RequestDidNotMatchRoutes();    await _next.Invoke(httpContext);  }  else  {    httpContext.Features[typeof(IRoutingFeature)] = new RoutingFeature()    {      RouteData = context.RouteData,    };    await context.Handler(context.HttpContext);  }}

首先,通過 httpContext 來初始化路由上下文(RouteContext),然后把用戶配置的路由規則添加到路由上下文RouteData中的Routers中去。

接下來 await _router.RouteAsync(context) , 就是用到了 IRouter 接口中的 RouteAsync 方法了。

我們接著跟蹤 RouteAsync 這個函數,看其內部都做了什么? 我們又跟蹤到了RouteCollection.cs 這個類:

我們看一下 RouteAsync 的流程:

public async virtual Task RouteAsync(RouteContext context){  var snapshot = context.RouteData.PushState(null, values: null, dataTokens: null);  for (var i = 0; i < Count; i++)  {    var route = this[i];    context.RouteData.Routers.Add(route);    try    {      await route.RouteAsync(context);      if (context.Handler != null)      {        break;      }    }    finally    {      if (context.Handler == null)      {        snapshot.Restore();      }    }  }}

我覺得這個類,包括函數設計的很巧妙,如果是我的話,我不一定能夠想的出來,所以我們通過看源碼也能夠學到很多新知識。

為什么說設計的巧妙呢? RouteCollection 繼承了 IRouter 但是并沒有具體的對路由進行處理,而是通過循環來重新將路由上下文分發的具體的路由處理程序上。我們來看一下他的流程:

1、為了提高性能,創建了一個RouteDataSnapshot 快照對象,RouteDataSnapshot是一個結構體,它存儲了 Route 中的路由數據信息。

2、循環當前 RouteCollection 中的 Router,添加到 RouterContext里的Routers中,然后把RouterContext交給Router來處理。

3、當沒有處理程序處理當前路由 snapshot.Restore() 重新初始化快照狀態。

接下來就要看具體的路由處理對象了,我們從 RouteBase 開始。

1、RouteBase 的構造函數會初始化 RouteTemplate, Name, DataTokens, Defaults.
 Defaults 是默認配置的路由參數。

2、RouteAsync 中會進行一系列檢查,如果沒有匹配到URL對應的路由就會直接返回。

3、使用路由參數匹配器 RouteConstraintMatcher 進行匹配,如果沒有匹配到,同樣直接返回。

4、如果匹配成功,會觸發 OnRouteMatched(RouteContext context)函數,它是一個抽象函數,具體實現位于 Route.cs 中。

然后,我們再繼續跟蹤到 Route.cs 中的 OnRouteMatch,一起來看一下:

protected override Task OnRouteMatched(RouteContext context){    context.RouteData.Routers.Add(_target);  return _target.RouteAsync(context);}

_target 值得當前路由的處理程序,那么具體是哪個路由處理程序呢? 我們一起探索一下。

我們知道,我們創建路由一共有MapRoute,MapGet,MapPost,MapPut,MapDelete,MapVerb... 等等這寫方式,我們分別對應說一下每一種它的路由處理程序是怎么樣的,下面是一個示例:

app.UseRouter(routes =>{  routes.DefaultHandler = new RouteHandler((httpContext) =>  {    var request = httpContext.Request;    return httpContext.Response.WriteAsync($"");  });            routes  .MapGet("api/get/{id}", (request, response, routeData) => {})  .MapMiddlewareRoute("api/middleware", (appBuilder) =>              appBuilder.Use((httpContext, next) => httpContext.Response.WriteAsync("Middleware!")           ))  .MapRoute(     name: "AllVerbs",     template: "api/all/{name}/{lastName?}",     defaults: new { lastName = "Doe" },     constraints: new { lastName = new RegexRouteConstraint(new Regex("[a-zA-Z]{3}",RegexOptions.CultureInvariant, RegexMatchTimeout)) });});

按照上面的示例解釋一下,

MapRoute:使用這種方式的話,必須要給 DefaultHandler 賦值處理程序,否則會拋出異常,通常情況下我們會使用RouteHandler類。

MapVerb: MapPost,MapPut 等等都和它類似,它將處理程序作為一個 RequestDelegate 委托提供了出來,也就是說我們實際上在自己處理HttpContext的東西,不會經過RouteHandler處理。

MapMiddlewareRoute:需要傳入一個 IApplicationBuilder 委托,實際上 IApplicationBuilder Build之后也是一個 RequestDelegate,它會在內部 new 一個 RouteHandler 類,然后調用的 MapRoute。

這些所有的矛頭都指向了 RouteHandler , 我們來看看 RouteHandler 吧。

public class RouteHandler : IRouteHandler, IRouter{  // ...略  public Task RouteAsync(RouteContext context)  {    context.Handler = _requestDelegate;    return TaskCache.CompletedTask;  }}

什么都沒干,僅僅是將傳入進來的 RequestDelegate 賦值給了 RouteContext 的處理程序。

最后,代碼會執行到 RouterMiddleware 類中的 Invoke 方法的最后一行 await context.Handler(context.HttpContext),這個時候開始調用Handler委托,執行用戶代碼。

總結

我們來總結一下以上流程:

首先傳入請求會到注冊的 RouterMiddleware 中間件,然后它 RouteAsync 按順序調用每個路由上的方法。當一個請求到來的時候,IRouter實例選擇是否處理已經設置到 RouteContext Handler 上的一個非空 RequestDelegate。如果Route已經為該請求設置處理程序,則路由處理會中止并且開始調用設置的Hanlder處理程序以處理請求。如果當前請求嘗試了所有路由都沒有找到處理程序的話,則調用next,將請求交給管道中的下一個中間件。

關于路由模板和參數約束源碼處理流程就不一一說了,有興趣可以直接看下源碼。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持VeVb武林網。


注:相關教程知識閱讀請移步到ASP.NET教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
亚洲丝袜av一区| 国产一区二区三区久久精品| 亚洲国产精品人久久电影| 欧美日韩在线免费| 久久久精品中文字幕| 久久香蕉频线观| 久久精品一本久久99精品| 91精品国产高清久久久久久久久| 91精品国产乱码久久久久久蜜臀| 国产精品成人一区二区| 亚洲电影免费观看高清完整版| 尤物九九久久国产精品的特点| 精品亚洲aⅴ在线观看| 欧美激情精品久久久久久蜜臀| 亚洲影视中文字幕| 国产精品视频网站| 久久影院中文字幕| 久久亚洲国产精品| 亚洲黄色www| 日韩中文字幕在线视频播放| 日韩成人在线观看| 96精品视频在线| 亚洲美女动态图120秒| 日韩三级成人av网| 岛国av一区二区三区| 久久精品久久久久| 欧美高清第一页| 欧美高清视频在线播放| 欧美老女人性生活| 久久久综合av| 久久久久久噜噜噜久久久精品| 97色在线观看免费视频| 欧美国产在线视频| 欧美在线国产精品| 黄色成人在线播放| 国产美女久久精品| 搡老女人一区二区三区视频tv| 亚洲精品久久久久国产| 91亚洲国产成人久久精品网站| 蜜臀久久99精品久久久无需会员| 91久久中文字幕| 欧美激情久久久| 在线视频日韩精品| 中文在线不卡视频| 欧美猛男性生活免费| 奇米成人av国产一区二区三区| 91精品国产91久久久久久不卡| 精品国产一区二区三区四区在线观看| 色777狠狠综合秋免鲁丝| 一色桃子一区二区| 久久久久久尹人网香蕉| 国产精品吊钟奶在线| 亚洲欧美一区二区激情| 波霸ol色综合久久| 欧美性黄网官网| 精品久久久久久亚洲精品| 久久这里只有精品视频首页| 日韩在线视频线视频免费网站| 精品久久中文字幕| 日韩美女在线观看一区| 欧美另类在线观看| 久久影视电视剧凤归四时歌| 欧美极品少妇全裸体| 中文字幕精品在线视频| 久久视频在线看| 日韩电影大全免费观看2023年上| 久久久免费av| 国产91在线高潮白浆在线观看| 91亚洲精品视频| 怡红院精品视频| 国产美女扒开尿口久久久| 91精品国产色综合久久不卡98| 久久久久久久影视| 欧美激情一区二区三级高清视频| 91久久久久久久久| 欧美极品美女视频网站在线观看免费| 亚洲热线99精品视频| 欧洲日本亚洲国产区| 久久精品国产亚洲7777| 国产精品视频导航| 久久久久久亚洲精品| 亚洲国产精品久久精品怡红院| 国产性猛交xxxx免费看久久| 国产伦精品一区二区三区精品视频| 综合国产在线视频| 国产精品美乳在线观看| 91在线直播亚洲| 亚洲激情成人网| 这里只有精品在线观看| 国产精品你懂得| 亚洲人成网7777777国产| 国产欧美日韩中文字幕在线| 久久亚洲精品一区二区| 精品欧美aⅴ在线网站| 午夜精品一区二区三区视频免费看| 午夜精品久久久久久久99黑人| 亚洲老头同性xxxxx| 欧美精品在线看| 91香蕉嫩草影院入口| 亚洲一级黄色片| 国产精品久久久久久久久久东京| 高清欧美性猛交xxxx| 亚洲va码欧洲m码| 中文字幕av一区中文字幕天堂| 91精品国产综合久久久久久蜜臀| 欧美性猛交xxxx免费看| 黑人欧美xxxx| 国产精品丝袜一区二区三区| 欧美裸体xxxx极品少妇软件| 亚洲欧美自拍一区| 欧美精品18videos性欧| 尤物精品国产第一福利三区| 久久国产视频网站| 亚洲va码欧洲m码| 精品中文字幕在线| 日韩精品在线私人| 国产精品一区二区久久久| 在线日韩中文字幕| 国产成人精品免费视频| 国产精品一区二区久久久| 黄色一区二区在线观看| 亚洲福利视频久久| 国产精品欧美日韩久久| 中文字幕精品一区久久久久| 欧美精品18videos性欧美| 亚洲一区二区三区乱码aⅴ| 欧美黑人xxxⅹ高潮交| 久久99精品久久久久久琪琪| 亚洲精品福利在线| 欧美日韩午夜激情| 成人高清视频观看www| 亚洲影院色在线观看免费| 97视频网站入口| 国产精品老女人视频| 正在播放欧美视频| 日韩一区二区福利| 亚洲天堂2020| 久久国产加勒比精品无码| 亚洲加勒比久久88色综合| 国产精品久久久久久久9999| 亚洲精品久久久久久久久久久| 日韩欧美在线免费观看| 亚洲直播在线一区| 亚洲免费中文字幕| 九九热这里只有在线精品视| 欧美视频专区一二在线观看| 国产精品亚洲激情| 国产一区二区久久精品| 久久久久一本一区二区青青蜜月| 国产成人福利网站| 日韩在线观看网址| 国产91色在线|| xxx欧美精品| 黄色成人av网| 久久视频免费在线播放| 色偷偷偷综合中文字幕;dd| 亚洲欧美日韩成人| 欧美精品videofree1080p| 17婷婷久久www| 北条麻妃一区二区三区中文字幕| 成人妇女淫片aaaa视频| 国产精品美女免费看| 日本韩国欧美精品大片卡二| 国产精品久久77777|