国产99久久精品_欧美日本韩国一区二区_激情小说综合网_欧美一级二级视频_午夜av电影_日本久久精品视频

最新文章專題視頻專題問答1問答10問答100問答1000問答2000關(guān)鍵字專題1關(guān)鍵字專題50關(guān)鍵字專題500關(guān)鍵字專題1500TAG最新視頻文章推薦1 推薦3 推薦5 推薦7 推薦9 推薦11 推薦13 推薦15 推薦17 推薦19 推薦21 推薦23 推薦25 推薦27 推薦29 推薦31 推薦33 推薦35 推薦37視頻文章20視頻文章30視頻文章40視頻文章50視頻文章60 視頻文章70視頻文章80視頻文章90視頻文章100視頻文章120視頻文章140 視頻2關(guān)鍵字專題關(guān)鍵字專題tag2tag3文章專題文章專題2文章索引1文章索引2文章索引3文章索引4文章索引5123456789101112131415文章專題3
問答文章1 問答文章501 問答文章1001 問答文章1501 問答文章2001 問答文章2501 問答文章3001 問答文章3501 問答文章4001 問答文章4501 問答文章5001 問答文章5501 問答文章6001 問答文章6501 問答文章7001 問答文章7501 問答文章8001 問答文章8501 問答文章9001 問答文章9501
當前位置: 首頁 - 科技 - 知識百科 - 正文

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

來源:懂視網(wǎng) 責編:小采 時間:2020-11-27 22:37:53
文檔

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

請求如何進入ASP.NET MVC框架:一、前言 對于WebForm開發(fā),請求通常是一個以.aspx結(jié)尾的url,對應一個物理文件,從代碼的角度來說它其實是一個控件(Page)。而在MVC中,一個請求對應的是一個Controller里的Action。熟悉asp.net的朋友都知道,asp.net請求實際都是交給HttpHan
推薦度:
導讀請求如何進入ASP.NET MVC框架:一、前言 對于WebForm開發(fā),請求通常是一個以.aspx結(jié)尾的url,對應一個物理文件,從代碼的角度來說它其實是一個控件(Page)。而在MVC中,一個請求對應的是一個Controller里的Action。熟悉asp.net的朋友都知道,asp.net請求實際都是交給HttpHan

一、前言

  對于WebForm開發(fā),請求通常是一個以.aspx結(jié)尾的url,對應一個物理文件,從代碼的角度來說它其實是一個控件(Page)。而在MVC中,一個請求對應的是一個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)部很多集合都用到了這種設計)。如下代碼:

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)
{
 //設置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的任務就是激活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。

聲明:本網(wǎng)頁內(nèi)容旨在傳播知識,若有侵權(quán)等問題請及時與本網(wǎng)聯(lián)系,我們將在第一時間刪除處理。TEL:177 7030 7066 E-MAIL:11247931@qq.com

文檔

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

請求如何進入ASP.NET MVC框架:一、前言 對于WebForm開發(fā),請求通常是一個以.aspx結(jié)尾的url,對應一個物理文件,從代碼的角度來說它其實是一個控件(Page)。而在MVC中,一個請求對應的是一個Controller里的Action。熟悉asp.net的朋友都知道,asp.net請求實際都是交給HttpHan
推薦度:
標簽: 請求 模型 框架
  • 熱門焦點

最新推薦

猜你喜歡

熱門推薦

專題
Top
主站蜘蛛池模板: 福利视频一区二区牛牛 | 一边摸一边叫床一边爽 | 久久91精品国产91久久 | 毛片国产 | 亚洲欧美日韩色图 | 亚洲国产成人综合精品2020 | 欧美另类在线观看 | 国产精品一区二区午夜嘿嘿嘿小说 | 成人欧美一区二区三区在线 | 日韩欧美国产电影 | 伊人精品成人久久综合欧美 | 久久久久国产成人精品亚洲午夜 | 久久久久免费精品国产小说 | 日韩1区| 亚洲欧美日韩国产色另类 | 日本一区二区三区在线观看 | 精品国产一区二区三区成人 | 国产一级成人毛片 | 午夜免费福利视频 | 中文字幕 自拍偷拍 | 国产在线视欧美亚综合 | 欧美日韩国产片 | 成人亚洲国产精品久久 | 日本一二三高清 | 国产精品一级片 | 国产精品亚洲片在线观看不卡 | 8x国产在线观看 | 一久久 | 国产精品久久久久… | 香蕉视频99 | 久久99精品国产99久久 | 日韩福利一区 | 国产色综合久久无码有码 | 熟年交尾五十路视频在线播放 | 久久久久国产成人精品亚洲午夜 | 免费啪啪网 | 国产一区亚洲二区三区 | 亚洲a∨精品一区二区三区下载 | 一边摸一边叫床一边爽 | 欧美色亚洲 | 久久久国产成人精品 |