Web流程-MVC执行

上一篇了解了 ISAPI执行原理,现在我们接着看 请求在MVC中是怎么处理的;

在了解MVC流程之前,我们先来了解一下一个重要的MvcHandler 类型,他是由路由生成的;

一. 路由模块

路由模块存在于 System.Web.Routing.dll 类库中,其中有一个重要的类 UrlRoutingModule;

  1. 网站配置主要分两种,全局和本站的Web.config配置;
    1.1. 全局配置目录D:\Windows\Microsoft.NET\Framework64\v4.0.30319\Config\web.config
<httpModules>
         .......
           <add name="UrlRoutingModule-4.0" type="System.Web.Routing.UrlRoutingModule" />
           .......
</httpModules>

1.2. 本站配置存在,应用程序的根目录下;

  1. 当请求管道接收到请求时,就会加载UrlRoutingModule,我们看内部怎么处理。我们看到在第七个管道事件PostResolveRequestCache中注册了一个事件;
 public class UrlRoutingModule : IHttpModule
    {
       
        protected virtual void Init(HttpApplication application)
        {
            if (application.Context.Items[_contextKey] == null)
            {
                application.Context.Items[_contextKey] = _contextKey;
                //第七个管道事件
                application.PostResolveRequestCache += new EventHandler(this.OnApplicationPostResolveRequestCache);
            }
        }
        
        private void OnApplicationPostResolveRequestCache(object sender, EventArgs e)
        {
            HttpApplication application = (HttpApplication) sender;
            HttpContextBase context = new HttpContextWrapper(application.Context);
            this.PostResolveRequestCache(context);
        }

        public virtual void PostResolveRequestCache(HttpContextBase context)
        {
//在RouteCollection中封装了所有的路由信息
            RouteData routeData = this.RouteCollection.GetRouteData(context);
            if (routeData != null)
            {
                // IRouteHandler 拿到具体的handle
                IRouteHandler routeHandler = routeData.RouteHandler;
                if (routeHandler == null)
                {
                    throw new InvalidOperationException(string.Format(CultureInfo.CurrentCulture, System.Web.SR.GetString("UrlRoutingModule_NoRouteHandler"), new object[0]));
                }
                if (!(routeHandler is StopRoutingHandler))
                {
                    RequestContext requestContext = new RequestContext(context, routeData);
                    context.Request.RequestContext = requestContext;
                    IHttpHandler httpHandler = routeHandler.GetHttpHandler(requestContext);
                    if (httpHandler == null)
                    {
                        throw new InvalidOperationException(string.Format(CultureInfo.CurrentUICulture, System.Web.SR.GetString("UrlRoutingModule_NoHttpHandler"), new object[] { routeHandler.GetType() }));
                    }
                    if (httpHandler is UrlAuthFailureHandler)
                    {
                        if (!FormsAuthenticationModule.FormsAuthRequired)
                        {
                            throw new HttpException(0x191, System.Web.SR.GetString("Assess_Denied_Description3"));
                        }
                        UrlAuthorizationModule.ReportUrlAuthorizationFailure(HttpContext.Current, this);
                    }
                    else
                    {
                        context.RemapHandler(httpHandler);
                    }
                }
            }
        }
......

2.1. 在Application_Strat方法 路由注册中通过方法MapRoute将请求的地址写入RouteCollection中。

public static Route MapRoute(this RouteCollection routes, string name, string url, object defaults, object constraints, string[] namespaces)
{
   ......
    Route item = new Route(url, new MvcRouteHandler()) {
        Defaults = CreateRouteValueDictionary(defaults),
        Constraints = CreateRouteValueDictionary(constraints),
        DataTokens = new RouteValueDictionary()
    };
    if ((namespaces != null) && (namespaces.Length > 0))
    {
        item.DataTokens["Namespaces"] = namespaces;
    }
    routes.Add(name, item);
    return item;
}

2.2. 通过方法 routeHandler.GetHttpHandler(requestContext) 实例MvcHandler的生成

  protected virtual IHttpHandler GetHttpHandler(RequestContext requestContext)
        {
            requestContext.HttpContext.SetSessionStateBehavior(this.GetSessionStateBehavior(requestContext));
            return new MvcHandler(requestContext);
        }

二.从MvcHandler的PR方法进入MVC流程的执行

一.Controller创建流程

以上介绍了MvcHandler在第七个事件中生成,在第11和12个事件之中调用PR方法开始响应请求,以下介绍PR中到底执行了什么;

  1. 通过ProcessRequestInit内部的工厂方法this.ControllerBuilder.GetControllerFactory().CreateController() 带入上下文对象创建Controller实例
 protected internal virtual void ProcessRequest(HttpContextBase httpContext)
        {
            IController controller;
            IControllerFactory factory;
            this.ProcessRequestInit(httpContext, out controller, out factory);
            try
            {
                controller.Execute(this.RequestContext);
            }
            finally
            {
                factory.ReleaseController(controller);
            }
        }

 private void ProcessRequestInit(HttpContextBase httpContext, out IController controller, out IControllerFactory factory)
        {
            HttpContext current = HttpContext.Current;
            ......
            factory = this.ControllerBuilder.GetControllerFactory();
            controller = factory.CreateController(this.RequestContext, requiredString);
           ......
        }
  1. 调用 controller.Execute(this.RequestContext)执行具体的Action; 通过ControllerBase的ExecuteCore 方法执行
  public abstract class ControllerBase : IController
    {
        ......
        protected virtual void Execute(RequestContext requestContext)
        {
            ......
            using (ScopeStorage.CreateTransientScope())
            {
                this.ExecuteCore();
            }
            ......
        }

3.调用Controller的ExecuteCore().GetRequiredString()方法;获取Action方法名称,并通过ExecuteCore().InvokeAction()触发.
3.1. 可以看到在InvokeAction.FindAction()方法中根据上下文,方法名 寻找对应的描述或特性;
3.2. 可以看到在InvokeAction.GetFilters()方法中加载过滤器

public abstract class Controller
{
protected override void ExecuteCore()
        {
            this.PossiblyLoadTempData();
            try
            {
                string requiredString = this.RouteData.GetRequiredString("action");
                if (!this.ActionInvoker.InvokeAction(base.ControllerContext, requiredString))
                {
                    this.HandleUnknownAction(requiredString);
                }
            }
            finally
            {
                this.PossiblySaveTempData();
            }
        }

public virtual bool InvokeAction(ControllerContext controllerContext, string actionName)
        {
            ......
            ControllerDescriptor controllerDescriptor = this.GetControllerDescriptor(controllerContext);
            ActionDescriptor actionDescriptor = this.FindAction(controllerContext, controllerDescriptor, actionName);
            if (actionDescriptor == null)
            {
                return false;
            }
            FilterInfo filters = this.GetFilters(controllerContext, actionDescriptor);
            try
            {
                AuthorizationContext context = this.InvokeAuthorizationFilters(controllerContext, filters.AuthorizationFilters, actionDescriptor);
          ......

  1. 通过InvokeActionMethodWithFilters方法得到了一个Result对象;我们具体来看先通过actionDescriptor.Execute()生成结果,最后转为 ActionResult对象返回
 protected virtual ActionExecutedContext InvokeActionMethodWithFilters(ControllerContext controllerContext, IList<IActionFilter> filters, ActionDescriptor actionDescriptor, IDictionary<string, object> parameters)
        {
            ActionExecutingContext preContext = new ActionExecutingContext(controllerContext, actionDescriptor, parameters);
            Func<ActionExecutedContext> seed = () 
=> new ActionExecutedContext(controllerContext, actionDescriptor, false, null) 
{ 
Result = this.InvokeActionMethod(controllerContext, actionDescriptor, parameters)
 };
            return filters.Reverse<IActionFilter>().Aggregate<IActionFilter, Func<ActionExecutedContext>>(seed, (next, filter) => () => InvokeActionMethodFilter(filter, preContext, next))();
        }

 protected virtual ActionResult InvokeActionMethod(ControllerContext controllerContext, ActionDescriptor actionDescriptor, IDictionary<string, object> parameters)
        {
            //此处执行action方法
            object actionReturnValue = actionDescriptor.Execute(controllerContext, parameters);
            return this.CreateActionResult(controllerContext, actionDescriptor, actionReturnValue);
        }

 protected virtual ActionResult CreateActionResult(ControllerContext controllerContext, ActionDescriptor actionDescriptor, object actionReturnValue)
        {
            ActionResult result;
            if (actionReturnValue == null)
            {
                return new EmptyResult();
            }
            //子类转基类返回
            if (actionReturnValue as ActionResult)
            {
                result = actionReturnValue as ActionResult;
            }
            else
            {
                ContentResult result2 = new ContentResult {
                    Content = Convert.ToString(actionReturnValue, CultureInfo.InvariantCulture)
                };
                result = result2;
            }
            return result;
        }
  1. 继续执行InvokeActionResultWithFilters方法完成Result的 AOP;
 protected virtual ResultExecutedContext InvokeActionResultWithFilters(ControllerContext controllerContext, IList<IResultFilter> filters, ActionResult actionResult)
        {
            ResultExecutingContext preContext = new ResultExecutingContext(controllerContext, actionResult);
            Func<ResultExecutedContext> seed = delegate {
                this.InvokeActionResult(controllerContext, actionResult);
                return new ResultExecutedContext(controllerContext, actionResult, false, null);
            };
            return filters.Reverse<IResultFilter>().Aggregate<IResultFilter, Func<ResultExecutedContext>>(seed, (next, filter) => () => InvokeActionResultFilter(filter, preContext, next))();
        }
二.View创建流程

ActionResult已经拿到,现在要对其进行渲染;我们看一下它内部的渲染方法

  1. 通过ActionResult基类ActionResultBase.ExecuteResult()方法,看到创建一个 ViewEngineResult渲染引擎
 public override void ExecuteResult(ControllerContext context)
        {
                 ......
            if (string.IsNullOrEmpty(this.ViewName))
            {
                this.ViewName = context.RouteData.GetRequiredString("action");
            }
            ViewEngineResult result = null;
            if (this.View == null)
            {
                result = this.FindView(context);
                this.View = result.View;
            }
  1. 通过ViewContext方法加载ViewData,TempData 等等数据生成ViewContext
 public override void ExecuteResult(ControllerContext context)
        {
          ......
            TextWriter output = context.HttpContext.Response.Output;
            ViewContext viewContext = new ViewContext(context, this.View, this.ViewData, this.TempData, output);
            this.View.Render(viewContext, output);
          ......
}


public ViewContext(ControllerContext controllerContext, IView view, ViewDataDictionary viewData, TempDataDictionary tempData, TextWriter writer) : base(controllerContext)
        {
            this._defaultFormContext = new System.Web.Mvc.FormContext();
            if (controllerContext == null)
            {
                throw new ArgumentNullException("controllerContext");
            }
            if (view == null)
            {
                throw new ArgumentNullException("view");
            }
            if (viewData == null)
            {
                throw new ArgumentNullException("viewData");
            }
            if (tempData == null)
            {
                throw new ArgumentNullException("tempData");
            }
            if (writer == null)
            {
                throw new ArgumentNullException("writer");
            }
            this.View = view;
            this.ViewData = viewData;
            this.Writer = writer;
            this.TempData = tempData;
        }
  1. 当上下文ViewContext,TextWriter已经准备好就可以渲染View了;我们从View.Render(viewContext, output)方法中看到,根据View类型生成类实例;
 public void Render(ViewContext viewContext, TextWriter writer)
        {
       ......
            Type compiledType = this.BuildManager.GetCompiledType(this.ViewPath);
            if (compiledType != null)
            {
                instance = this.ViewPageActivator.Create(this._controllerContext, compiledType);
            }
      ......
            this.RenderView(viewContext, writer, instance);
        }
  1. 我们看到从RenderView方法中生成WebViewPage 对象,并将返回数据写入对象属性中,最后调用ExecutePageHierarchy渲染;至此整个View渲染流程就结束了;
 protected override void RenderView(ViewContext viewContext, TextWriter writer, object instance)
        {
            if (writer == null)
            {
                throw new ArgumentNullException("writer");
            }
            WebViewPage page = instance as WebViewPage;
            if (page == null)
            {
                throw new InvalidOperationException(string.Format(CultureInfo.CurrentCulture, MvcResources.CshtmlView_WrongViewBase, new object[] { base.ViewPath }));
            }
            page.OverridenLayoutPath = this.LayoutPath;
            page.VirtualPath = base.ViewPath;
            page.ViewContext = viewContext;
            page.ViewData = viewContext.ViewData;
            page.InitHelpers();
            if (this.VirtualPathFactory != null)
            {
                page.VirtualPathFactory = this.VirtualPathFactory;
            }
            if (this.DisplayModeProvider != null)
            {
                page.DisplayModeProvider = this.DisplayModeProvider;
            }
            WebPageRenderingBase startPage = null;
            if (this.RunViewStartPages)
            {
                startPage = this.StartPageLookup(page, RazorViewEngine.ViewStartFileName, this.ViewStartFileExtensions);
            }
            HttpContextBase httpContext = viewContext.HttpContext;
            WebPageRenderingBase base4 = null;
            object model = null;
            page.ExecutePageHierarchy(new WebPageContext(httpContext, base4, model), writer, startPage);
        }

参考文档 :
博客园:[http://www.cnblogs.com/edisonchou/p/4226558.html]
博客园:[http://www.cnblogs.com/yinzixin/archive/2012/12/05/2799459.html]

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,837评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,551评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,417评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,448评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,524评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,554评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,569评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,316评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,766评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,077评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,240评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,912评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,560评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,176评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,425评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,114评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,114评论 2 352

推荐阅读更多精彩内容