Controller的激活
ASP.NET MVC的路由系统通过注册的路由表对当前HTTP请求实施路由解析,从而得到一个用于封装路由数据的RouteData对象,这个过程是通过自定义的UrlRoutingModule对HttpApplication的PostResolveRequestCache事件进行注册实现的。由于得到的RouteData对象中已经包含了目标Controller的名称,我们需要根据该名称激活对应的Controller对象。
1.MvcRouteHandler
通过前面的介绍我们知道,继承自RouteBase的Route类型具有一个类型为IRouteHandler借口的属性RouteHandler,它主要的用途就是用于根据指定的请求上下文(RequestContext)来获取一个HttpHandler对象。当GetRouteData方法被执行后,Route的RouteHandler属性值将反映在得到的RouteData的同名属性上,在默认的情况下,Route的RouteHandler属性是一个MvcRouteHandler对象。
对于我们的“迷你版”ASP.NET MVC框架来说,MvcRouteHandler是一个具有如下定义的类型。在实现的GetHttpHandler方法中它会直接返回一个MvcHandler对象。
2.MvcHandler
整个ASP.NET MVC框架是通过自定义的HttpModule和HttpHandler对ASP.NET进行扩展构建起来的,这个自定义的HttpModule类型就是UrlRouteModule,而这个自定义的HttpHandler类型则是需要重点介绍的MvcHandler。
UrlRouteModule在利用路由表对当前请求实施路由解析并得到封装路由数据的RouteData对象后,会调用其RouteHandler的GetHttpHandler方法得到一个HttpHandler对象,然后将其映射到当前的HTTP上下文。由于RouteData的RouteHandler来源于对应Route对象的RouteHandler,而后者在默认的情况下是一个MvcRouteHandler对象,所有默认情况下用于处理HTTP请求的就是这么一个MvcHandler对象。MvcHandler实现了对Controller对象的激活和对目标Action方法的执行。
如下面的代码片段所示,MvcHandler具有一个类型为RequestContext的属性,它表示当前请求上下文,该属性在构造函数中指定。MvcHandler在ProcessRequest方法中实现了对Controller对象的激活和执行。
3.Controller与ControllerFactory
我们为Controller定义了一个接口IController。如下所示,该接口具有唯一的方法Execute表示对当前Controller对象的执行。该方法在MvcHandler的ProcessRequest方法中被调用,而传入该方法的参数是表示当前请求上下文的RequestContext对象。
从MvcHandler的定义可以看到,Controller对象的激活是通过工厂模式实现的,我们为激活Controller的工厂定义了IControllerFactory接口。该接口具有唯一的方法CreateController,该方法根据当前请求上下文和通过路由解析得到的目标Controller的名称激活相应的Controller对象。
在MvcHandler的ProcessRequest方法中,他通过ControllerBuilder的静态属性Current得到当前的ControllerBuilder对象,并调用其GetControllerFactory方法获得当前的ControllerFactory。接下来MvcHandler通过从RequestContext中提取的RequestData对象获得目标Controller的名称,最后将它连同RequestContext一起作为参数调用ControllerFactory的CreateController方法实现对目标Controller对象的创建。
ControllerBuilder的整个定义如下,表示当前ControllerBuilder的静态只读属性Current在静态构造函数中被创建,其SetControllerFactory和GetControllerFactory方法用于ControllerFactory的注册和获取。
在之前我们建立在自定义ASP.NET MVC框架的Web应用,我们就是通过当前的ControllerBuilder来注册ControllerFactory。注册的ControllerFactory类型为DefaultControllerFactory。
作为默认ControllerFactory的DefaultControllerFactory类型定义如下。由于激活Controller对象的前提是能够正确的解析出Controller的真实类型,作为CreateController方法输入参数的controllerName仅仅表示Controller的名称,所以我们需要加上Controller字符后缀作为类型名称。在DefaultControllerFactory类型被加载的时候(静态构造函数被调用),他通过BuilderMessage加载所有被引用的程序集,得到所有实现了接口的IController的类型并将其缓存起来。在CreateController方法中,DefaultControllerFactory根据Controller的名称从保存的Controller类型列表中得到对应的Controller类型,并通过反射的方式创建它。
上面我们详细地介绍了 Controller 的激活原理,现在将关注点返回到 Controller 自身。我们通过实现 IController 接口为所有的 Controller 定义了一个具有如下定义的 ControllerBase 抽象基类,从中可以看到在实现的 Execute 方法中 ControllerBase 通过一个实现了接口 IActionInvoker 的对象完成了针对 Action 方法的执行。
在下一章我们讲解Action的执行。