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

主頁 > 知識庫 > 請求如何進入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框架的相關信息資訊供網民參考!
  • 推薦文章
    校园春色亚洲色图_亚洲视频分类_中文字幕精品一区二区精品_麻豆一区区三区四区产品精品蜜桃
    欧美精品三级日韩久久| 亚洲精品一线二线三线无人区| 欧美日韩中文字幕一区二区| 亚洲精品免费在线| 欧美性色综合网| 日本不卡一二三| 久久网站热最新地址| 成+人+亚洲+综合天堂| 一区二区三区四区在线播放| 制服丝袜激情欧洲亚洲| 国产91精品一区二区麻豆网站| 一区二区高清在线| 久久久久久99精品| 欧美午夜视频网站| 国产精品乡下勾搭老头1| 夜夜嗨av一区二区三区中文字幕| 正在播放亚洲一区| 97超碰欧美中文字幕| 欧美bbbbb| 夜夜揉揉日日人人青青一国产精品| 欧美精品在线观看播放| av不卡免费电影| 激情综合色丁香一区二区| 国产精品久久久久三级| 在线综合亚洲欧美在线视频| 91在线一区二区三区| 国内成人精品2018免费看| 亚洲自拍偷拍av| 国产精品美女一区二区在线观看| 精品视频999| 成人福利在线看| 韩国精品一区二区| 日本欧美久久久久免费播放网| 综合精品久久久| 久久蜜桃一区二区| 欧美一级片在线| 欧美中文字幕一区二区三区亚洲| 粉嫩蜜臀av国产精品网站| 久久99九九99精品| 日本欧洲一区二区| 亚洲成av人在线观看| 亚洲视频图片小说| 精品区一区二区| 欧美一区午夜精品| 69精品人人人人| 欧美老女人第四色| 欧美网站大全在线观看| 91国在线观看| 在线精品亚洲一区二区不卡| av电影在线观看一区| 99精品国产热久久91蜜凸| 99热99精品| 色综合久久99| 欧美日韩精品一区二区三区四区 | 中文字幕不卡的av| 久久精品网站免费观看| 久久精品人人做人人爽97| 精品久久久久久久久久久院品网 | 成人午夜看片网址| 成人网在线播放| 99精品视频免费在线观看| eeuss鲁片一区二区三区 | 亚洲欧美国产毛片在线| 亚洲色欲色欲www| 一区二区三区美女| 亚洲成人av电影| 蜜臀av性久久久久蜜臀aⅴ| 韩国在线一区二区| 91网站在线播放| 欧美一区二区视频观看视频| 精品国产在天天线2019| 日本一区二区三区国色天香| 亚洲美女在线一区| 日本伊人色综合网| 高潮精品一区videoshd| 色婷婷av一区二区三区之一色屋| 欧美影院一区二区| 欧美xxxxx牲另类人与| 日本一区二区三区久久久久久久久不| 国产精品黄色在线观看| 亚洲不卡av一区二区三区| 精品亚洲porn| 色婷婷综合五月| 亚洲精品一区二区三区福利| 亚洲精品视频在线| 国产毛片精品国产一区二区三区| 成人一区二区三区中文字幕| 欧美色视频在线观看| 久久综合久久综合久久| 亚洲免费av高清| 精品亚洲aⅴ乱码一区二区三区| 不卡的av电影| 亚洲精品在线电影| 亚洲成人免费视频| 99久久精品国产观看| 欧美大白屁股肥臀xxxxxx| 国产精品久久久久久久第一福利 | yourporn久久国产精品| 91精品欧美一区二区三区综合在| 亚洲国产精品成人综合色在线婷婷 | 91精品国产综合久久精品麻豆| 国产女主播在线一区二区| 日韩av在线播放中文字幕| 99久久综合狠狠综合久久| 精品国产一区二区三区忘忧草| 亚洲色图在线播放| 成人免费观看视频| 久久精品视频一区二区| 日韩精品每日更新| 在线免费观看日韩欧美| 国产精品久久久久一区二区三区| 久久精品国产澳门| 欧美一级黄色片| 丝袜美腿亚洲综合| 欧美午夜宅男影院| 亚洲一区二区免费视频| 91麻豆精品在线观看| 亚洲国产精品精华液ab| 国产精品夜夜爽| 久久久久国产精品厨房| 国产一区二区三区香蕉 | 欧美国产日韩一二三区| 国产一区在线观看视频| 欧美成人乱码一区二区三区| 日本在线不卡一区| 日韩欧美高清dvd碟片| 蜜桃视频一区二区三区在线观看| 欧美精品高清视频| 毛片一区二区三区| 欧美v国产在线一区二区三区| 五月天一区二区| 欧美成人性福生活免费看| 精品一区二区三区在线播放视频| 日韩欧美国产综合一区| 免费不卡在线观看| www国产成人免费观看视频 深夜成人网| 老司机免费视频一区二区三区| 日韩免费高清av| 国产精品456| 亚洲视频在线观看一区| 欧美视频在线播放| 久久超碰97中文字幕| 久久久国际精品| 一本一道久久a久久精品 | 亚洲美女视频在线观看| 色噜噜久久综合| 亚洲高清免费观看高清完整版在线观看| 欧美做爰猛烈大尺度电影无法无天| 亚洲成av人片在www色猫咪| 91麻豆精品国产91久久久资源速度 | 亚洲成人激情av| 欧美成人精品二区三区99精品| 激情综合网av| 亚洲精品免费播放| 91精品蜜臀在线一区尤物| 国产黑丝在线一区二区三区| 国产精品免费视频观看| 欧美一区二区三区系列电影| 成人精品免费看| 亚洲综合男人的天堂| 久久久精品国产免费观看同学| 91一区二区在线观看| 美女精品一区二区| 亚洲精品中文字幕乱码三区| 日韩一级片在线观看| 一本到三区不卡视频| 精品亚洲aⅴ乱码一区二区三区| 亚洲人成小说网站色在线| 日韩精品中文字幕一区二区三区| jizzjizzjizz欧美| 激情综合色综合久久综合| 亚洲成人免费在线| 最近日韩中文字幕| 久久久午夜精品理论片中文字幕| 欧美性videosxxxxx| 成人丝袜18视频在线观看| 日本中文字幕一区二区视频| 亚洲免费在线视频| 欧美激情在线一区二区三区| 3d成人h动漫网站入口| 色88888久久久久久影院野外| 国产精品乡下勾搭老头1| 美国毛片一区二区三区| 五月综合激情网| 亚洲制服丝袜av| 综合色天天鬼久久鬼色| 国产精品区一区二区三区| 国产亚洲一区二区在线观看| 欧美一区二区三区影视| 7777精品伊人久久久大香线蕉| 日本道在线观看一区二区| 99久久99久久免费精品蜜臀| 国产综合成人久久大片91| 蜜臀久久99精品久久久久宅男| 亚洲成人777| 日日摸夜夜添夜夜添亚洲女人| 亚洲国产中文字幕| 一区二区三区日韩| 玉米视频成人免费看| 亚洲理论在线观看|