ASP.NETCoreMVC之Routing路由的示例分析-創(chuàng)新互聯(lián)

這篇文章主要為大家展示了“ASP.NET Core MVC之Routing路由的示例分析”,內(nèi)容簡而易懂,條理清晰,希望能夠幫助大家解決疑惑,下面讓小編帶領(lǐng)大家一起研究并學(xué)習(xí)一下“ASP.NET Core MVC之Routing路由的示例分析”這篇文章吧。

成都創(chuàng)新互聯(lián)服務(wù)項目包括江寧網(wǎng)站建設(shè)、江寧網(wǎng)站制作、江寧網(wǎng)頁制作以及江寧網(wǎng)絡(luò)營銷策劃等。多年來,我們專注于互聯(lián)網(wǎng)行業(yè),利用自身積累的技術(shù)優(yōu)勢、行業(yè)經(jīng)驗、深度合作伙伴關(guān)系等,向廣大中小型企業(yè)、政府機構(gòu)等提供互聯(lián)網(wǎng)行業(yè)的解決方案,江寧網(wǎng)站推廣取得了明顯的社會效益與經(jīng)濟效益。目前,我們服務(wù)的客戶以成都為中心已經(jīng)輻射到江寧省份的部分城市,未來相信會繼續(xù)擴大服務(wù)區(qū)域并繼續(xù)獲得客戶的支持與信任!

路由(Routing)功能介紹

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

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

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

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

  1. URL 匹配

  2. URL 生成

  3. IRouter 接口

  4. 路由模板

  5. 模板約束

Getting Started

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

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

Microsoft.AspNetCore.Routing.Abstractions

大致看完整個結(jié)構(gòu)之后,我可能發(fā)現(xiàn)了幾個關(guān)鍵的接口,理解了這幾個接口的作用后能夠幫助我們在后續(xù)的閱讀中事半功倍。

IRouter

Microsoft.AspNetCore.Routing.Abstractions 中有一個關(guān)鍵的接口就是 IRouter:


public interface IRouter
{
 Task RouteAsync(RouteContext context);

 VirtualPathData GetVirtualPath(VirtualPathContext context);
}

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

IRouteHandler

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


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

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

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

RouteData,封裝了當(dāng)前路由中的數(shù)據(jù)信息,它包含三個主要屬性,分別是DataTokens,Routers,Values

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

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

Values: 當(dāng)前路由的路徑下包含的鍵值。

還有一個RouteValueDictionary, 它是一個集合類,主要是用來存放路由中的一些數(shù)據(jù)信息的,沒有直接使用IEnumerable<KeyValuePair<string, string>>這個數(shù)據(jù)結(jié)構(gòu)是應(yīng)為它的內(nèi)部存儲轉(zhuǎn)換比較復(fù)雜,它的構(gòu)造函數(shù)接收一個 Object 的對象,它會嘗試將Object 對象轉(zhuǎn)化為自己可以識別的集合。

IRoutingFeature

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

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

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

IRouteConstraint

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

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

/// 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的一個主要實現(xiàn),我們閱讀代碼的時候可以從它的入口開始閱讀。

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


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

public static IApplicationBuilder UseRouter(this IApplicationBuilder builder, IRouter router)
{
  //...略
  
 return builder.UseMiddleware<RouterMiddleware>(router);
}

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

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

Routing 中間件的入口是RouterMiddleware這個類,通過這個中間件注冊到 Http 的管道處理流程中, ASP.NET Core MVC 會把它默認的作為其配置項的一部分,當(dāng)然你也可以把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),然后把用戶配置的路由規(guī)則添加到路由上下文RouteData中的Routers中去。

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

我們接著跟蹤RouteAsync這個函數(shù),看其內(nèi)部都做了什么? 我們又跟蹤到了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();
      }
    }
  }
}

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

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

1、為了提高性能,創(chuàng)建了一個RouteDataSnapshot 快照對象,RouteDataSnapshot是一個結(jié)構(gòu)體,它存儲了 Route 中的路由數(shù)據(jù)信息。

2、循環(huán)當(dāng)前 RouteCollection 中的 Router,添加到 RouterContext里的Routers中,然后把RouterContext交給Router來處理。

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

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

1、RouteBase 的構(gòu)造函數(shù)會初始化RouteTemplate,Name,DataTokens,Defaults.
 Defaults 是默認配置的路由參數(shù)。

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

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

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

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


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

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

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

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的東西,不會經(jīng)過RouteHandler處理。

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

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

public class RouteHandler : IRouteHandler, IRouter
{
  // ...略

  public Task RouteAsync(RouteContext context)
  {
    context.Handler = _requestDelegate;
    return TaskCache.CompletedTask;
  }
}

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

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

以上是“ASP.NET Core MVC之Routing路由的示例分析”這篇文章的所有內(nèi)容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內(nèi)容對大家有所幫助,如果還想學(xué)習(xí)更多知識,歡迎關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道!

當(dāng)前名稱:ASP.NETCoreMVC之Routing路由的示例分析-創(chuàng)新互聯(lián)
網(wǎng)站地址:http://muchs.cn/article18/ceepgp.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供靜態(tài)網(wǎng)站、App開發(fā)、全網(wǎng)營銷推廣、手機網(wǎng)站建設(shè)、網(wǎng)站維護品牌網(wǎng)站建設(shè)

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)

成都網(wǎng)站建設(shè)公司