ASP.NET MVC Controller激活系统详解2
一.引言
此篇博文紧接上篇博文进行阐述,本篇博文阐述的主题是Controller激活和url路由
二.总述
ASP.NET路由系统是HTTP请求抵达服务端的第一道屏障,它根据注册的路由规则对拦截的请求进行匹配并解析包含目标的Controller和Action名称的路由信息。当前ControllerBuilder具有用于激活Controller对象的ControllerFactory。
ASP.NET路由系统的核心是一个叫做UrlRoutingModule的自定义HttpModule,路由的实现是它通过注册代表当前web应用的HttpApplication的PostResolverRequestCache事件对HttpHandler的动态映射来实现的。具体来说,它通过以RouteTable的静态属性Routes代表的全局路由表对请求进行匹配并得到一个RouteData对象。RouteData具有一个实现了接口IRouteHandler属性RouteHandler,通过该属性的GetHttpHandler得到最终被映射到当前请求的HttpHandler。
三.MVC
对于MVC应用来说,RouteData的RouteHandler属性类型为MvcRouteHandler,体现在MvcRouteHandler类型上关于HttpHandler的提供机制基本上(不是完全等同)可以通过如下的代码来表示。MvcRouteHandler维护着一个ControllerFactory对象,该对象可以在构造函数中指定,如果没有显示指定则直接通过调用当前ControllerBuilder的GetControllerFactory来获取。
1 public class MvcRouteHandler:IRouteHandler 2 { 3 private IControllerFactory _controllerFactory; 4 6 public MvcRouteHandler() { } 7 public MvcRouteHandler(IControllerFactory controllerFactory) 8 { 9 _controllerFactory = controllerFactory; 10 } 11 12 protected virtual SessionStateBehavior GetSessionStateBehavior(RequestContext context) 13 { 14 string str = (string)context.RouteData.Values["controller"]; //RouteData.Values获取路由的url参数值和默认值的集合 15 if(string.IsNullOrWhiteSpace(str)) 16 { 17 throw new InvalidOperationException("MvcRouteHandler_RouteValuesHasNoController"); 18 } 19 IControllerFactory factory = _controllerFactory ?? ControllerBuilder.Current.GetControllerFactory(); 20 return factory.GetControllerSessionBehavior(context, str); 21 } 22 public virtual IHttpHandler GetHttpHandler(RequestContext context) 23 { 24 context.HttpContext.SetSessionStateBehavior(GetSessionStateBehavior(context)); 25 return new MvcHandler(context); 26 } 27 }
用于提供HttpHandler的GetHttpHandler方法中,除了返回一个实现了IHttpHandler接口的MvcHandler对象之外,还需要对当前Http上下文的会话状态行为模式进行设置。具体来说,首先通过包含在传入RequestContext的RouteData对象得到Controller的名称,该名称连同RequestContext对象一起传入ControllerFactory的GetControllerSessionBehavior方法得到一个类型为SessionStateBehavior的枚举。最后通过RequestContext得到表示当前HTTP上下文的HttpContextBase对象(实际上是一个HttpContextWrapper对象)并调用其SetSessionStateBehavior方法。
我们知道RouteData中的RouteHandler属性最初来源于对应的Route对象的同名属性,而当我们调用RouteCollection的扩展方法MapRoute方法时,其内部会直接创建并添加一个Route对象。由于在创建Route对象是并没有显式指定ControllerFactory,所以通过当前ControllerBuilder的GetControllerFactory方法得到的ControllerFactory默认被使用。
通过当前ControllerBuilder的GetControllerFactory方法得到的ControllerFactory仅仅用于获取会话状态行为模式,而MvcHandler真正将它用于创建Controller。MvcHandler中关于对请求处理的逻辑基本上可以通过如下的代码片断来体现。如下面的代码片断所示,MvcHandler具有一个表示当前请求上下文的RequestContext属性,该属性在构造函数中被初始化。