校园春色亚洲色图_亚洲视频分类_中文字幕精品一区二区精品_麻豆一区区三区四区产品精品蜜桃

主頁 > 知識庫 > 請求如何進入ASP.NET MVC框架

請求如何進入ASP.NET MVC框架

熱門標簽:地圖定位圖標標注 塔城代理外呼系統 天心智能電銷機器人 地圖標注的公司有哪些 400電話辦理哪家性價比高 濮陽外呼電銷系統怎么樣 遂寧市地圖標注app 代理接電話機器人如何取消 地圖標注專業團隊

一、前言

  對于WebForm開發,請求通常是一個以.aspx結尾的url,對應一個物理文件,從代碼的角度來說它其實是一個控件(Page)。而在MVC中,一個請求對應的是一個Controller里的Action。熟悉asp.net的朋友都知道,asp.net請求實際都是交給HttpHandler處理(實現了IHttpHandler的類型)。無論是.aspx,.ashx,.asmx 還是MVC里的Action,請求都會交給HttpHandler。具體是在管道事件中,會根據請求創建一個HttpHandler,并執行它的PR方法。對于aspx和ashx都很好理解,因為它們本身就實現了IHttpHandler接口,而MVC的Controller和Action都和HttpHandler沒有關系,它是如何實現的呢?接下來我們就看一個請求是如何進入mvc框架內部的。

二、例子

  WebForm和MVC都是建立在asp.net平臺上的,Webform出現得比較早,那么MVC是如何做到在不影響底層框架,實現擴展的呢?這主要得益于asp.net的路由機制。路由機制并不屬于MVC,WebForm也可以使用它。它的目的是讓一個請求與物理文件分離,原理是通過映射關系,將請求映射到指定的HttpHandler。例如我們也可以將一個/Admin/User.aspx?name=張三 的請求映射成可讀性更好的/Admin/張三。下面是兩種url的注冊方式:

public static void RegisterRoutes(RouteCollection routes)
{
  //MVC
  routes.MapRoute(
    name: "Default",
    url: "{controller}/{action}/{id}",
    defaults: new { controller = "Home", action = "Index", id = UrlParameter.Optional }
  );
 
  //WebForm
  routes.MapPageRoute(
    routeName: "WebForm",
    routeUrl: "Admin/{user}",
    physicalFile: "~/Admin/User.aspx"
  );
}

  RouteCollection是一個Route集合,Route封裝了名稱、url模式、約束條件、默認值等路由相關信息。其中,MapPageRoute是RouteCollection定義的方法,而MapRoute是MVC擴展出來的(擴展方法的好處就是可以在不修改原有代碼的情況下添加所需的功能)。它們的目的都是一樣的,創建一個Route對象,添加到集合當中;我們也可以new 一個Route對象,然后調用RouteCollection.Add,效果是一樣的。下面我們主要關注MVC的實現過程,WebForm其實也是類似的。

三、分析源碼

  接下來我們看MVC是如何利用路由機制實現擴展的。路由機制是通過一個UrlRoutingModule完成的,它是一個實現了IHttpModule的類,路由模塊已經默認幫我們注冊好了。HttpModule通過注冊HttpApplication事件參與到管道處理請求中,具體是訂閱HttpApplication某個階段的事件。路由機制就是利用這個原理,UrlRoutingModule訂閱了PostResolveRequestCache 事件,實現url的映射。為什么是該事件呢?因為該事件的下一步就要完成請求和物理文件的映射,所以必須要此之前進行攔截。核心代碼如下:

public class UrlRoutingModule : IHttpModule {
  public RouteCollection RouteCollection {
    get {
      if (_routeCollection == null) {
        //全局的RouteCollection集合
        _routeCollection = RouteTable.Routes;
      }
      return _routeCollection;
    }
    set {
      _routeCollection = value;
    }
  }
 
  protected virtual void Init(HttpApplication application) {
    //注冊PostResolveRequestCache事件
    application.PostResolveRequestCache += OnApplicationPostResolveRequestCache;
  }
 
  private void OnApplicationPostResolveRequestCache(object sender, EventArgs e) {
    //創建上下文
    HttpApplication app = (HttpApplication)sender;
    HttpContextBase context = new HttpContextWrapper(app.Context);
    PostResolveRequestCache(context);
  }
 
  public virtual void PostResolveRequestCache(HttpContextBase context) {
    //1.獲取RouteData
    RouteData routeData = RouteCollection.GetRouteData(context);
    if (routeData == null) {
      return;
    }
    //2.獲取IRouteHandler
    IRouteHandler routeHandler = routeData.RouteHandler;
    if (routeHandler == null) {
       
    }
     
    //RequestContext保證了HttpContext和RouteData,在后續使用
    RequestContext requestContext = new RequestContext(context, routeData);
 
    context.Request.RequestContext = requestContext;
 
    //3.獲取IHttpHandler
    IHttpHandler httpHandler = routeHandler.GetHttpHandler(requestContext);
 
    //重新映射到處理程序
    context.RemapHandler(httpHandler);
  }
}  

  我們關注主要方法PostResolveRequestCache,這里有三個關鍵步驟。

步驟一. 獲取RouteData

  RouteData是對Route的包裝,在后續的處理中使用。它的獲取是通過RouteCollection獲得的,這個和上面注冊用到的RouteTable.Routes是同一個集合對象。調用RouteCollection的GetRouteData會遍歷它的每一個項,也就是Route對象,然后調用Route對象的GetRouteData方法(MVC內部很多集合都用到了這種設計)。如下代碼:

public RouteData GetRouteData(HttpContextBase httpContext) {
  using (GetReadLock()) {
    foreach (RouteBase route in this) {
      RouteData routeData = route.GetRouteData(httpContext);
      if (routeData != null) {           
        return routeData;
      }
    }
  }
  return null;
}

  Route對象的GetRouteData方法如下:

public override RouteData GetRouteData(HttpContextBase httpContext) {
  string requestPath = httpContext.Request.AppRelativeCurrentExecutionFilePath.Substring(2) + httpContext.Request.PathInfo;
 
  //結合默認值,匹配url
  RouteValueDictionary values = _parsedRoute.Match(requestPath, Defaults);
 
  if (values == null) {
    return null;
  }
 
  //包裝成RouteData,這里為什么不放在if后面呢?
  RouteData routeData = new RouteData(this, RouteHandler);
 
  //匹配約束
  if (!ProcessConstraints(httpContext, values, RouteDirection.IncomingRequest)) {
    return null;
  }
 
  //RouteData的Values和DataTokens都來自于Route
  foreach (var value in values) {
    routeData.Values.Add(value.Key, value.Value);
  }
  if (DataTokens != null) {
    foreach (var prop in DataTokens) {
      routeData.DataTokens[prop.Key] = prop.Value;
    }
  }
 
  return routeData;
}

  可以看到,Route對象的GetRouteData方法會匹配url模式,和檢查約束條件,如何不符合會返回null。如果匹配,則new一個RouteData。

步驟二、獲取IRouteHandler接口對象

  上面創建RouteData,參數分別是當前Route對象和它的RouteHandler屬性。RouteHandler是一個IRouteHandler,這是一個重要接口,它的定義如下:

public interface IRouteHandler {
  IHttpHandler GetHttpHandler(RequestContext requestContext);
}

  很明顯,它是用于獲取IHttpHandler的。那么Route對象的RouteHandler屬性又是在哪里初始化的呢?我們回到開始的注冊方法,routes.MapRoute,這個方法根據傳遞的參數創建一個Route對象,該方法的實現如下:

public static Route MapRoute(this RouteCollection routes, string name, string url, object defaults, object constraints, string[] namespaces)
{
  //創建一個Route對象,它的IRouteHandler為MvcRouteHandler
  Route route = new Route(url, new MvcRouteHandler())
  {
    Defaults = CreateRouteValueDictionary(defaults),
    Constraints = CreateRouteValueDictionary(constraints),
    DataTokens = new RouteValueDictionary()
  };
 
  if ((namespaces != null)  (namespaces.Length > 0))
  {
    route.DataTokens["Namespaces"] = namespaces;
  }
 
  //將Route注冊到RouteCollection中
  routes.Add(name, route);
 
  return route;
}

  在創建Route時,除了傳遞url模式外,還默認幫我們傳遞了一個MvcRouteHandler,它實現了IRouteHandler接口。
步驟三、獲取IHttpHandler接口對象

  有了MvcRouteHandler,就可以調用它的GetHttpHandler方法獲取IHttpHandler了,該方法實現如下:

protected virtual IHttpHandler GetHttpHandler(RequestContext requestContext)
{
  //設置session狀態
  requestContext.HttpContext.SetSessionStateBehavior(GetSessionStateBehavior(requestContext));
 
  //返回一個實現了IHttpHandler的MvcHandler
  return new MvcHandler(requestContext);
}

  可以看到,它返回了一個MvcHandler,MvcHandler就實現了IHttpHandler接口。所以開頭說的,請求本質都是交給HttpHandler的,其實MVC也是這樣的,請求交給了MvcHandler處理。我們可以看MvcHandler定義和主要方法:

public class MvcHandler : IHttpAsyncHandler, IHttpHandler, IRequiresSessionState
{
   protected internal virtual IAsyncResult BeginProcessRequest(HttpContextBase httpContext, AsyncCallback callback, object state)
  {
    IController controller;
    IControllerFactory factory;
 
    //這個方法里會激活Controller對象
    ProcessRequestInit(httpContext, out controller, out factory);
 
    IAsyncController asyncController = controller as IAsyncController;
    if (asyncController != null)
    {
      // asynchronous controller
      BeginInvokeDelegate beginDelegate = delegate(AsyncCallback asyncCallback, object asyncState)
      {
        try
        {
          //調用Controller的BeginExecute方法
          return asyncController.BeginExecute(RequestContext, asyncCallback, asyncState);
        }
        catch
        {
          factory.ReleaseController(asyncController);
          throw;
        }
      };
 
      EndInvokeDelegate endDelegate = delegate(IAsyncResult asyncResult)
      {
        try
        {
          asyncController.EndExecute(asyncResult);
        }
        finally
        {
          factory.ReleaseController(asyncController);
        }
      };
 
      SynchronizationContext syncContext = SynchronizationContextUtil.GetSynchronizationContext();
      AsyncCallback newCallback = AsyncUtil.WrapCallbackForSynchronizedExecution(callback, syncContext);
      return AsyncResultWrapper.Begin(newCallback, state, beginDelegate, endDelegate, _processRequestTag);
    }
    else
    {
      // synchronous controller
      Action action = delegate
      {
        try
        {
          controller.Execute(RequestContext);
        }
        finally
        {
          factory.ReleaseController(controller);
        }
      };
 
      return AsyncResultWrapper.BeginSynchronous(callback, state, action, _processRequestTag);
    }
  }
}

  可以看到,MvcHandler的任務就是激活Controller,并執行它的Execute方法。這個過程和Webform里的頁面處理是很相似的,.aspx請求到來,會根據虛擬路徑找到實現IHttpHandler的Page(類似于路由機制根據url模式找到MvcHandler),然后進入Page的頁面周期(類似于Mvc的激活Controller,然后執行Action過程)。

四、總結

接下來,簡單總結一下請求進入到MVC框架的過程:

1.添加路由對象Route到全局的RouteCollection,Route的IRouteHandler初始化為MvcRouteHandler。

2. UrlRoutingModule注冊 HttpApplication PostResolveRequestCache事件,實現請求攔截。
3. 請求到來, 在處理事件中遍歷RouteCollection,調用每一個Route對象的GetRouteData獲取RouteData包裝對象。

4. 調用MvcRouteHandler的GetHttpHandler獲取MvcHandler。

5. 調用HttpContext的RemapHandler將請求映射到MvcHandler處理程序。

6. 執行MvcHandler的PR方法,激活Controller,執行Action。

以上就是本文的全部內容,希望對大家的學習有所幫助。

您可能感興趣的文章:
  • 擴展ASP.NET MVC三層框架且使用StructureMap實現依賴注入1-Model層
  • ASP.NET MVC+EF框架+EasyUI實現權限管系列
  • 使用ASP.NET.4.5.1+MVC5.0 搭建一個包含 Ninject框架 項目
  • 支持ASP.NET MVC、WebFroM的表單驗證框架ValidationSuar使用介紹
  • ASP.NET MVC5網站開發項目框架(二)
  • ASP.NET MVC5 網站開發框架模型、數據存儲、業務邏輯(三)

標簽:汕頭 婁底 重慶 河南 吉林 麗江 宜春 本溪

巨人網絡通訊聲明:本文標題《請求如何進入ASP.NET MVC框架》,本文關鍵詞  請求,如何,進入,ASP.NET,MVC,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《請求如何進入ASP.NET MVC框架》相關的同類信息!
  • 本頁收集關于請求如何進入ASP.NET MVC框架的相關信息資訊供網民參考!
  • 推薦文章
    校园春色亚洲色图_亚洲视频分类_中文字幕精品一区二区精品_麻豆一区区三区四区产品精品蜜桃
    成人av午夜电影| 91久久精品一区二区三| 91精品国产麻豆| 男女视频一区二区| 国产视频一区二区在线| 色婷婷精品久久二区二区蜜臀av | 中文字幕中文字幕一区二区| 91蜜桃传媒精品久久久一区二区| 亚洲专区一二三| 日韩一二三四区| 成人性生交大片免费看中文网站| 一区二区三区中文在线| 91精品国产乱| av中文字幕不卡| 亚洲国产日韩av| 久久久噜噜噜久噜久久综合| 一本大道综合伊人精品热热| 日韩综合小视频| 欧美国产精品久久| 91精品国产aⅴ一区二区| 成人国产精品免费观看| 亚洲第一精品在线| 日本一二三不卡| 欧美乱熟臀69xxxxxx| 国产精品自在在线| 亚洲成人午夜影院| 国产精品美女一区二区在线观看| 7777精品伊人久久久大香线蕉 | 亚洲高清不卡在线| 中文字幕第一区| 欧美日韩一区二区三区四区五区| 国产精品一区二区91| 一区二区三区在线视频播放| 日本一区二区在线不卡| 欧美va天堂va视频va在线| 欧美日韩中文字幕一区| 91精品福利视频| 春色校园综合激情亚洲| 国产成人在线观看免费网站| 精品一区二区免费在线观看| 三级久久三级久久久| 亚洲一区二区三区视频在线播放 | 中文字幕欧美国产| 欧美大片一区二区| 51精品秘密在线观看| 欧美三级乱人伦电影| 99re这里只有精品视频首页| 成人短视频下载| 成人一区二区三区| 丁香五精品蜜臀久久久久99网站 | 91精品国产色综合久久不卡蜜臀| 欧美日韩一区二区电影| 欧美体内she精高潮| 欧美日精品一区视频| 欧美午夜精品久久久久久超碰| 色噜噜夜夜夜综合网| 91免费观看在线| 色噜噜久久综合| 欧美影视一区在线| 欧美日韩综合不卡| 欧美久久高跟鞋激| 欧美一区二区在线播放| 日韩一区二区影院| 日韩欧美国产精品| 久久蜜桃一区二区| 日本一区二区三区高清不卡| 欧美激情一区三区| 亚洲视频中文字幕| 一区二区高清在线| 亚洲狠狠爱一区二区三区| 亚洲v日本v欧美v久久精品| 日韩专区欧美专区| 国产麻豆一精品一av一免费| 国产iv一区二区三区| 成人国产免费视频| 精品视频在线免费观看| 日韩一区二区在线观看视频播放| www国产精品av| 国产精品国产三级国产普通话三级| 国产精品国产三级国产专播品爱网| 一区二区在线观看不卡| 亚洲成av人片观看| 久久成人免费网站| 成人a免费在线看| 欧美私人免费视频| 亚洲精品在线电影| 日韩美女精品在线| 午夜不卡av免费| 国产一区二区三区电影在线观看| 国产不卡视频在线观看| 色婷婷久久久久swag精品| 欧美一区二区三区公司| 国产精品婷婷午夜在线观看| 亚洲福利视频导航| 国产91在线观看| 欧美视频在线播放| 久久精品夜色噜噜亚洲aⅴ| 夜色激情一区二区| 狠狠色丁香九九婷婷综合五月| 成人不卡免费av| 这里是久久伊人| 亚洲欧洲另类国产综合| 亚洲超丰满肉感bbw| 国产91精品欧美| 欧美色网一区二区| 中文字幕 久热精品 视频在线| 亚洲成av人片在线| 成人的网站免费观看| 3d成人动漫网站| 国产精品毛片久久久久久久| 香蕉影视欧美成人| 成人h动漫精品| 欧美va日韩va| 一级中文字幕一区二区| 国内欧美视频一区二区 | 国产三级久久久| 午夜私人影院久久久久| 国产精品系列在线观看| 欧美男男青年gay1069videost| 国产欧美日韩在线看| 三级影片在线观看欧美日韩一区二区| 不卡视频免费播放| 欧美精品一区二区三区高清aⅴ| 亚洲精品中文在线| 国产福利一区二区三区在线视频| 欧日韩精品视频| 国产精品乱人伦| 国产一区欧美二区| 欧美久久久久免费| 一区二区三区影院| 国产a精品视频| 欧美videos中文字幕| 丝袜亚洲另类欧美综合| 91成人看片片| 中文字幕中文字幕一区二区| 国产精品夜夜嗨| 日韩欧美卡一卡二| 亚洲国产成人av好男人在线观看| 国产91精品精华液一区二区三区| 欧美大度的电影原声| 日韩国产欧美视频| 欧美精品丝袜久久久中文字幕| 一二三四区精品视频| 91色porny蝌蚪| 亚洲日本丝袜连裤袜办公室| 成人自拍视频在线| 国产日韩欧美亚洲| 夫妻av一区二区| 久久久精品日韩欧美| 精品一区二区免费看| 欧美一区二区三区在| 日韩在线一区二区三区| 欧美一区二区三区播放老司机| 偷拍日韩校园综合在线| 91精品欧美福利在线观看| 婷婷成人激情在线网| 欧美精品18+| 男女男精品视频网| 日韩一区二区麻豆国产| 婷婷综合五月天| 精品视频在线免费| 亚洲国产精品久久久男人的天堂| 成人av在线一区二区| 国产精品乱码久久久久久 | 97久久精品人人做人人爽50路| 久久精品无码一区二区三区| 精品无人码麻豆乱码1区2区| 欧美久久久久久久久中文字幕| 亚洲成a天堂v人片| 欧美日韩国产另类一区| 亚洲国产日韩精品| 欧美区一区二区三区| 日日夜夜免费精品| 欧美影院一区二区| 亚洲第一成人在线| 欧美日韩国产首页在线观看| 亚洲.国产.中文慕字在线| 欧美日韩亚洲综合一区二区三区| 亚洲尤物视频在线| 欧美日韩视频在线第一区| 国产精品对白交换视频| 色婷婷精品久久二区二区蜜臀av| 亚洲欧美另类图片小说| 在线视频一区二区免费| 亚洲午夜精品网| 欧美福利视频一区| 蜜臀av性久久久久蜜臀aⅴ流畅| 欧美一区二区三区播放老司机| 免费视频最近日韩| 久久久综合网站| 成人av在线一区二区| 亚洲精选视频免费看| 欧美日本韩国一区二区三区视频 | 美女精品自拍一二三四| 中文av一区二区| 欧洲精品中文字幕| 日本中文一区二区三区| 精品国产3级a| www.欧美日韩| 亚洲国产成人高清精品|