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

                

 

posted @ 2018-10-19 19:25  小王子的博客  阅读(298)  评论(0编辑  收藏  举报