学习ASP.NET MVC5框架揭秘笔记-ASP.NET MVC是如何运行的(三)
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对象。
public class MvcRouteHandler : IRouteHandler { public IHttpHandler GetHttpHandler(RequestContext requestContext) { return new MvcHandler(requestContext); } }
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对象的激活和执行。
public class MvcHandler : IHttpHandler { public bool IsReusable { get { return false; } } public RequestContext RequestContext { get; private set; } public MvcHandler(RequestContext requestContext) { this.RequestContext = requestContext; } public void ProcessRequest(HttpContext context) { string controllerName = this.RequestContext.RouteData.Controller; IControllerFactory controllerFactory = ControllerBuilder.Current.GetControllerFactory(); IController controller = controllerFactory.CreateController(this.RequestContext, controllerName); controller.Execute(this.RequestContext); } }
3.Controller与ControllerFactory
我们为Controller定义了一个接口IController。如下所示,该接口具有唯一的方法Execute表示对当前Controller对象的执行。该方法在MvcHandler的ProcessRequest方法中被调用,而传入该方法的参数是表示当前请求上下文的RequestContext对象。
public interface IController { void Execute(RequestContext requestContext); }
从MvcHandler的定义可以看到,Controller对象的激活是通过工厂模式实现的,我们为激活Controller的工厂定义了IControllerFactory接口。该接口具有唯一的方法CreateController,该方法根据当前请求上下文和通过路由解析得到的目标Controller的名称激活相应的Controller对象。
public interface IControllerFactory { IController CreateController(RequestContext requestContext, string controllerName); }
在MvcHandler的ProcessRequest方法中,他通过ControllerBuilder的静态属性Current得到当前的ControllerBuilder对象,并调用其GetControllerFactory方法获得当前的ControllerFactory。接下来MvcHandler通过从RequestContext中提取的RequestData对象获得目标Controller的名称,最后将它连同RequestContext一起作为参数调用ControllerFactory的CreateController方法实现对目标Controller对象的创建。
ControllerBuilder的整个定义如下,表示当前ControllerBuilder的静态只读属性Current在静态构造函数中被创建,其SetControllerFactory和GetControllerFactory方法用于ControllerFactory的注册和获取。
public class ControllerBuilder { private Func factoryThunk; public static ControllerBuilder Current { get; private set; } static ControllerBuilder() { Current = new ControllerBuilder(); } public IControllerFactory GetControllerFactory() { return factoryThunk(); } public void SetControllerFactory(IControllerFactory controllerFactory) { factoryThunk = () => controllerFactory; } }
在之前我们建立在自定义ASP.NET MVC框架的Web应用,我们就是通过当前的ControllerBuilder来注册ControllerFactory。注册的ControllerFactory类型为DefaultControllerFactory。
public class Global : System.Web.HttpApplication { protected void Application_Start(object sender, EventArgs e) { ControllerBuilder.Current.SetControllerFactory(new DefaultControllerFactory()); } }
作为默认ControllerFactory的DefaultControllerFactory类型定义如下。由于激活Controller对象的前提是能够正确的解析出Controller的真实类型,作为CreateController方法输入参数的controllerName仅仅表示Controller的名称,所以我们需要加上Controller字符后缀作为类型名称。在DefaultControllerFactory类型被加载的时候(静态构造函数被调用),他通过BuilderMessage加载所有被引用的程序集,得到所有实现了接口的IController的类型并将其缓存起来。在CreateController方法中,DefaultControllerFactory根据Controller的名称从保存的Controller类型列表中得到对应的Controller类型,并通过反射的方式创建它。
public class DefaultControllerFactory : IControllerFactory { private static List controllerTypes = new List(); static DefaultControllerFactory() { foreach (Assembly assembly in BuildManager.GetReferencedAssemblies()) { foreach (Type type in assembly.GetTypes().Where(type => typeof(IController).IsAssignableFrom(type))) { controllerTypes.Add(type); } } } public IController CreateController(RequestContext requestContext,string controllerName) { string typeName = controllerName + Controller; Type controllerType = controllerTypes.FirstOrDefault(c => string.Compare(typeName, c.Name, true) == 0); if (null == controllerType) { return null; } return (IController)Activator.CreateInstance(controllerType); } }
上面我们详细地介绍了Controller的激活原理,现在将关注点返回到Controller自身。我们通过实现IController接口为所有的Controller定义了一个具有如下定义的ControllerBase抽象基类,从中可以看到在实现的Execute方法中ControllerBase通过一个实现了接口IActionInvoker的对象完成了针对Action方法的执行。
public abstract class ControllerBase : IController { protected IActionInvoker ActionInvoker { get; set; } public ControllerBase() { this.ActionInvoker = new ControllerActionInvoker(); } public void Execute(RequestContext requestContext) { ControllerContext context = new ControllerContext { RequestContext = requestContext, Controller = this }; string actionName = requestContext.RouteData.ActionName; this.ActionInvoker.InvokeAction(context, actionName); } }
推荐阅读
-
学习ASP.NET MVC5框架揭秘笔记-ASP.NET MVC是如何运行的(三)
-
学习ASP.NET MVC5框架揭秘笔记-ASP.NET MVC路由(三)
-
学习ASP.NET MVC5框架揭秘笔记-ASP.NET MVC是如何运行的(二)
-
学习ASP.NET MVC框架揭秘笔记-实例演示:SC模式的应用
-
ASP.NET MVC是如何运行的[1]: 建立在“伪”MVC框架上的Web应用
-
学习ASP.NET MVC5框架揭秘笔记-ASP.NET MVC是如何运行的(四)
-
学习ASP.NET MVC5框架揭秘笔记-ASP.NET MVC路由
-
学习ASP.NET MVC5框架揭秘笔记-ASP.NET MVC路由(三)
-
学习ASP.NET MVC5框架揭秘笔记-ASP.NET MVC路由(二)
-
学习ASP.NET MVC5框架揭秘笔记-ASP.NET MVC是如何运行的(一)