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来获取。

     public class MvcRouteHandler:IRouteHandler
{
private IControllerFactory _controllerFactory; public MvcRouteHandler() { }
public MvcRouteHandler(IControllerFactory controllerFactory)
{
_controllerFactory = controllerFactory;
} protected virtual SessionStateBehavior GetSessionStateBehavior(RequestContext context)
{
string str = (string)context.RouteData.Values["controller"]; //RouteData.Values获取路由的url参数值和默认值的集合
if(string.IsNullOrWhiteSpace(str))
{
throw new InvalidOperationException("MvcRouteHandler_RouteValuesHasNoController");
}
IControllerFactory factory = _controllerFactory ?? ControllerBuilder.Current.GetControllerFactory();
return factory.GetControllerSessionBehavior(context, str);
}
public virtual IHttpHandler GetHttpHandler(RequestContext context)
{
context.HttpContext.SetSessionStateBehavior(GetSessionStateBehavior(context));
return new MvcHandler(context);
}
}

用于提供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属性,该属性在构造函数中被初始化。

上一篇:bzoj 4557: [JLoi2016]侦察守卫 树归


下一篇:【锋利的JQuery-学习笔记】切换网页皮肤-且保存于Cookie