您的位置:首页 > 编程语言 > ASP

学习ASP.NET MVC5框架揭秘笔记-ASP.NET MVC是如何运行的(三)

2015-05-15 22:29 791 查看


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对象。

[html] view
plaincopy





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对象的激活和执行。

[html] view
plaincopy





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对象。

[html] view
plaincopy





public interface IController  

    {  

        void Execute(RequestContext requestContext);  

    }  

从MvcHandler的定义可以看到,Controller对象的激活是通过工厂模式实现的,我们为激活Controller的工厂定义了IControllerFactory接口。该接口具有唯一的方法CreateController,该方法根据当前请求上下文和通过路由解析得到的目标Controller的名称激活相应的Controller对象。

[html] view
plaincopy





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的注册和获取。

 

[html] view
plaincopy





public class ControllerBuilder  

    {  

        private Func<IControllerFactory> 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。

[html] view
plaincopy





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类型,并通过反射的方式创建它。

[html] view
plaincopy





public class DefaultControllerFactory : IControllerFactory  

    {  

        private static List<Type> controllerTypes = new List<Type>();  

   

        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方法的执行。

[html] view
plaincopy





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);  

        }  

    }  

在下一章我们讲解Action的执行。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: