一、前言
對于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); }}
新聞熱點
疑難解答
圖片精選