婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av

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

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

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

一、前言

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

二、例子

  WebForm和MVC都是建立在asp.net平臺上的,Webform出現(xiàn)得比較早,那么MVC是如何做到在不影響底層框架,實現(xiàn)擴展的呢?這主要得益于asp.net的路由機制。路由機制并不屬于MVC,WebForm也可以使用它。它的目的是讓一個請求與物理文件分離,原理是通過映射關(guān)系,將請求映射到指定的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模式、約束條件、默認值等路由相關(guān)信息。其中,MapPageRoute是RouteCollection定義的方法,而MapRoute是MVC擴展出來的(擴展方法的好處就是可以在不修改原有代碼的情況下添加所需的功能)。它們的目的都是一樣的,創(chuàng)建一個Route對象,添加到集合當中;我們也可以new 一個Route對象,然后調(diào)用RouteCollection.Add,效果是一樣的。下面我們主要關(guān)注MVC的實現(xiàn)過程,WebForm其實也是類似的。

三、分析源碼

  接下來我們看MVC是如何利用路由機制實現(xiàn)擴展的。路由機制是通過一個UrlRoutingModule完成的,它是一個實現(xiàn)了IHttpModule的類,路由模塊已經(jīng)默認幫我們注冊好了。HttpModule通過注冊HttpApplication事件參與到管道處理請求中,具體是訂閱HttpApplication某個階段的事件。路由機制就是利用這個原理,UrlRoutingModule訂閱了PostResolveRequestCache 事件,實現(xiàn)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) {
    //創(chuàng)建上下文
    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,在后續(xù)使用
    RequestContext requestContext = new RequestContext(context, routeData);
 
    context.Request.RequestContext = requestContext;
 
    //3.獲取IHttpHandler
    IHttpHandler httpHandler = routeHandler.GetHttpHandler(requestContext);
 
    //重新映射到處理程序
    context.RemapHandler(httpHandler);
  }
}  

  我們關(guān)注主要方法PostResolveRequestCache,這里有三個關(guān)鍵步驟。

步驟一. 獲取RouteData

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

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;
 
  //結(jié)合默認值,匹配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接口對象

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

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

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

public static Route MapRoute(this RouteCollection routes, string name, string url, object defaults, object constraints, string[] namespaces)
{
  //創(chuàng)建一個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;
}

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

  有了MvcRouteHandler,就可以調(diào)用它的GetHttpHandler方法獲取IHttpHandler了,該方法實現(xiàn)如下:

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

  可以看到,它返回了一個MvcHandler,MvcHandler就實現(xiàn)了IHttpHandler接口。所以開頭說的,請求本質(zhì)都是交給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
        {
          //調(diào)用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的任務(wù)就是激活Controller,并執(zhí)行它的Execute方法。這個過程和Webform里的頁面處理是很相似的,.aspx請求到來,會根據(jù)虛擬路徑找到實現(xiàn)IHttpHandler的Page(類似于路由機制根據(jù)url模式找到MvcHandler),然后進入Page的頁面周期(類似于Mvc的激活Controller,然后執(zhí)行Action過程)。

四、總結(jié)

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

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

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

4. 調(diào)用MvcRouteHandler的GetHttpHandler獲取MvcHandler。

5. 調(diào)用HttpContext的RemapHandler將請求映射到MvcHandler處理程序。

6. 執(zhí)行MvcHandler的PR方法,激活Controller,執(zhí)行Action。

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

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

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

巨人網(wǎng)絡(luò)通訊聲明:本文標題《請求如何進入ASP.NET MVC框架》,本文關(guān)鍵詞  請求,如何,進入,ASP.NET,MVC,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請?zhí)峁┫嚓P(guān)信息告之我們,我們將及時溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《請求如何進入ASP.NET MVC框架》相關(guān)的同類信息!
  • 本頁收集關(guān)于請求如何進入ASP.NET MVC框架的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章
    婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av
    亚洲欧洲制服丝袜| 欧美亚洲国产bt| 国产99精品视频| 欧美成人精品福利| 亚洲乱码日产精品bd| 成人一区二区三区| 久久嫩草精品久久久精品一| 久久国产精品72免费观看| 精品久久久久久久久久久院品网| 日韩av在线播放中文字幕| 精品国产伦一区二区三区观看方式| 日本成人中文字幕| 欧美性xxxxx极品少妇| 日韩高清一级片| 日韩欧美国产系列| 国产成人鲁色资源国产91色综| 精品久久久久久久久久久久久久久 | 日本亚洲欧美天堂免费| 欧美日韩视频在线第一区| 亚洲成av人片一区二区梦乃| 色欲综合视频天天天| 精彩视频一区二区| 欧美精品一卡二卡| 69堂精品视频| 狠狠色综合日日| 中文字幕二三区不卡| 欧美色爱综合网| 波多野结衣精品在线| 亚洲已满18点击进入久久| 精品免费日韩av| 欧美日韩mp4| 91国偷自产一区二区开放时间| 国产最新精品精品你懂的| 亚洲1区2区3区4区| 亚洲日本丝袜连裤袜办公室| 久久久久久日产精品| 欧美日韩在线三级| 亚洲国产精品自拍| 91亚洲大成网污www| 亚洲国产精品成人综合| 亚洲午夜成aⅴ人片| 久久免费视频色| 欧美人妖巨大在线| 色狠狠一区二区三区香蕉| 国产不卡视频一区二区三区| 亚洲成人综合视频| 亚洲人123区| 亚洲欧美在线高清| 欧美激情一区二区三区| 欧美电影免费观看完整版| 欧美三级电影在线看| 欧美性一二三区| 成人av在线一区二区三区| 免费视频最近日韩| 无码av免费一区二区三区试看| 国产精品久久久久影院| 中文字幕的久久| 欧美成人性福生活免费看| 91精品福利在线一区二区三区| 色美美综合视频| 99视频有精品| 成人综合在线观看| 国产精品一区二区三区99| 国产伦精品一区二区三区视频青涩 | 欧美综合色免费| 色婷婷av一区二区三区大白胸 | 91在线视频播放地址| 成人免费va视频| 色诱视频网站一区| 欧美探花视频资源| 欧美日韩三级一区二区| 欧美精品日韩综合在线| 欧美一卡二卡在线观看| 日韩欧美亚洲一区二区| 日韩欧美国产一区二区三区| 久久综合精品国产一区二区三区 | 国产欧美在线观看一区| 精品精品欲导航| 久久婷婷综合激情| 日本一区二区免费在线| 欧美韩国日本不卡| 中文字幕国产一区二区| 亚洲人成人一区二区在线观看 | 美女网站视频久久| 国产成人精品三级| 欧美中文字幕不卡| 久久精品亚洲乱码伦伦中文| 一区二区三区免费网站| 黄色精品一二区| 色噜噜狠狠色综合欧洲selulu| 日韩视频一区二区三区在线播放| 欧美国产激情二区三区| 亚洲成年人网站在线观看| 国产精品亚洲一区二区三区在线| 在线精品观看国产| 国产亚洲成年网址在线观看| 亚洲最新在线观看| 成人一区在线看| 日韩精品一区二区三区视频播放| 欧美日本一道本在线视频| 久久麻豆一区二区| 亚洲大片一区二区三区| gogo大胆日本视频一区| 精品欧美久久久| 亚洲成人免费av| 91视频精品在这里| 国产欧美精品国产国产专区| 日本不卡中文字幕| 欧美午夜宅男影院| 亚洲摸摸操操av| 成人精品高清在线| 久久先锋影音av| 久久国产精品露脸对白| 91精品国产综合久久久久久久久久 | 欧美在线观看禁18| 国产精品私人自拍| 国产不卡在线一区| 久久久国产精品不卡| 麻豆精品国产传媒mv男同| 欧美二区三区91| 午夜国产精品一区| 欧美日韩国产一二三| 亚洲一区二区三区国产| 欧洲一区二区av| 亚洲小少妇裸体bbw| 欧美色电影在线| 天天av天天翘天天综合网色鬼国产| 91国在线观看| 亚洲高清不卡在线| 欧美乱妇23p| 男人的j进女人的j一区| 欧美一卡二卡三卡| 国内精品国产三级国产a久久 | 亚洲精品午夜久久久| 91年精品国产| 一区二区三区精品在线观看| 欧日韩精品视频| 天堂va蜜桃一区二区三区| 777亚洲妇女| 麻豆传媒一区二区三区| 久久青草国产手机看片福利盒子| 国产一区二区剧情av在线| 中文字幕乱码久久午夜不卡| 99国产一区二区三精品乱码| 一卡二卡三卡日韩欧美| 91麻豆精品国产| 国产麻豆精品一区二区| 亚洲少妇中出一区| 91麻豆精品国产91久久久久久| 久久99深爱久久99精品| 国产精品久久一卡二卡| 欧美性xxxxx极品少妇| 久久99精品一区二区三区 | 麻豆视频一区二区| 国产午夜久久久久| 91片黄在线观看| 美腿丝袜亚洲色图| 中文字幕人成不卡一区| 在线看不卡av| 激情综合一区二区三区| 国产精品你懂的在线| 欧美日韩国产成人在线91| 国产不卡在线一区| 午夜精品久久久久久久久久久| 久久亚区不卡日本| 欧美在线视频全部完| 国产成人av电影免费在线观看| 一区2区3区在线看| 国产无人区一区二区三区| 欧美亚一区二区| 丁香六月综合激情| 免费成人小视频| 亚洲一区电影777| 久久婷婷国产综合精品青草| 欧美午夜精品久久久| 成人黄色大片在线观看| 久久99精品久久久久久动态图 | 国产精品资源在线观看| 亚洲一区二区影院| 国产精品麻豆视频| 精品1区2区在线观看| 欧美三级韩国三级日本一级| 盗摄精品av一区二区三区| 奇米精品一区二区三区四区 | 在线综合亚洲欧美在线视频| 国产精品一级在线| 麻豆精品一区二区三区| 亚洲亚洲精品在线观看| 1000部国产精品成人观看| 久久在线免费观看| 91精品国产综合久久久久久| 97精品国产露脸对白| 国产成人精品一区二区三区四区| 国产精品久久久久久久久免费樱桃| 欧美疯狂做受xxxx富婆| 欧美三级三级三级| 欧美专区日韩专区| 在线观看www91| 欧美性感一区二区三区| 91视频一区二区三区|