前面 的篇章, 解析了Action方法的查找, 以及 Authorize, Action, Result, Error 过滤器的加载时机. 也花了两篇去看授权和错误过滤器的使用. 但是对于 Action/Result 的执行以及Action/Result里面的两个过滤器的执行时机, 并没有清晰看到.

一、Action 方法的执行和过滤器的执行

  1. //System.Web.Mvc.ControllerActionInvoker
  2. protected virtual ActionExecutedContext InvokeActionMethodWithFilters(ControllerContext controllerContext,
       IList<IActionFilter> filters, ActionDescriptor actionDescriptor, IDictionary<string, object> parameters)
  3. {
  4. ActionExecutingContext preContext = new ActionExecutingContext(controllerContext, actionDescriptor, parameters);
  5. //这里定义了一个Func委托, 返回值为 ActionExecutedContext 类型
       //在返回的 ActionExecutedContext 类型实例中, 对 Result 指定了执行方法
       //这里其实都是在指定待执行的方法, 在它执行完毕后, 并没有任何的方法被执行. 以下标红的部分, 就是执行 Action 方法
       Func<ActionExecutedContext> seed = () => new ActionExecutedContext(controllerContext, actionDescriptor, false, null)
        { Result = this.InvokeActionMethod(controllerContext, actionDescriptor, parameters) };
  6. return filters
           //filters变量是一个以controller为首元素的List<IActionFilter>集合,这里的目的就是要反转集合, 变成
           // IActionFilter->Controller
          .Reverse<IActionFilter>()
           //聚合函数, 这里的 next 其实就是 seed 对象
           //filter 是 IEnumberable<IActionFilter> 中的元素
           //这里的效果, 就是返回累加后的 seed 对象
          .Aggregate<IActionFilter, Func<ActionExecutedContext>>(seed,
             (next, filter) => () => InvokeActionMethodFilter(filter, preContext, next))
          (); //这一步执行方法
  7. }

这里的代码就是 前面 的 3.4, 只是我对他进行了分割, 这样容易看懂点

这里有一个 Aggregate<>() 可能不是那么好理解, 先上一个demo, 应该是能够促进理解的.

  1. var numbers = new int[] { , , , };
  2. // 使用当前总值乘以下一个数值得到新的总值
  3. var product = numbers.Aggregate((total, next) => total * next);
  4. 最后的值是: X X X =

这里的执行过程是这样的:

1). 当第一次执行 Aggregate() 的时候, 匿名方法(next, filter)中的next 就是 seed 对象, filter就是ActionFilter对象. 他将InvokeActionMethodFilter(filter, preContext, next)作为返回值, 并作为下一次执行(next, filter)中的next参数

2). 第二次执行(next, filter)的时候, next参数的值是 InvokeActionMethodFilter(filter, preContext, next) , filter是Controller, 最终将

InvokeActionMethodFilter(filter, preContext, next)最为结果返回给 filters 变量.

整个执行表达式为  : filters = InvokeActionMethodFilter(controller, preContext, InvokeActionMethodFilter(ActionFilter, preContext, seed));

经过上述步骤后, filters形成了一个链表, 其最先执行的还是Controller.

1. InvokeActionMethodFilter

  1. internal static ActionExecutedContext InvokeActionMethodFilter(IActionFilter filter,
      ActionExecutingContext preContext, Func<ActionExecutedContext> continuation)
  2. {
       //这一步执行 ActionExecuting 过滤器
  3. filter.OnActionExecuting(preContext);
  4. if (preContext.Result != null)
  5. {
         //当不能通过过滤器的时候, 会走这里
  6. return new ActionExecutedContext(preContext, preContext.ActionDescriptor, true, null)
          { Result = preContext.Result };
  7. }
  8. bool flag = false;
  9. ActionExecutedContext filterContext = null;
  10. try
  11. {
         //1.第一次执行这一步,会跳转到 InvokeActionFilter(filter, preContext, next)中去执行
          //2.第二次运行到这里的时候,会去执行 上面的 seed 方法
          //在执行seed方法的时候, 就回去运行 InvokeActionMethod() 方法
  12. filterContext = continuation();
  13. }
  14. catch (ThreadAbortException)
  15. {
  16. filterContext = new ActionExecutedContext(preContext, preContext.ActionDescriptor, false, null);
  17. filter.OnActionExecuted(filterContext);
  18. throw;
  19. }
  20. catch (Exception exception)
  21. {
  22. flag = true;
  23. filterContext = new ActionExecutedContext(preContext, preContext.ActionDescriptor, false, exception);
  24. filter.OnActionExecuted(filterContext);
  25. if (!filterContext.ExceptionHandled)
  26. {
  27. throw;
  28. }
  29. }
  30. if (!flag)
  31. {
  32. filter.OnActionExecuted(filterContext);
  33. }
  34. return filterContext;
  35. }

在这里, 已经能看到 ActionExecuting 和 ActionExecuted 两个过滤器的执行时机了

根据上面的注释, 然后看一下 InvokeActionMethod 方法

  1. protected virtual ActionResult InvokeActionMethod(ControllerContext controllerContext,
         ActionDescriptor actionDescriptor, IDictionary<string, object> parameters)
  2. {
  3. object actionReturnValue = actionDescriptor.Execute(controllerContext, parameters);
  4. return this.CreateActionResult(controllerContext, actionDescriptor, actionReturnValue);
  5. }

这里的 Execute 方法, 返回的就是 ActionResult了, 调用CreateActionResult 是为了进行类型转换, 将返回的actionReturnValue 转换为ActionResult类型.

这里其实就是执行的 Action 方法. 并返回执行结果.

二、Result 的执行和过滤器的执行

  1. //ControllerActionInvoker
  2. protected virtual ResultExecutedContext InvokeActionResultWithFilters(ControllerContext controllerContext,
       IList<IResultFilter> filters, ActionResult actionResult)
  3. {
  4. ResultExecutingContext preContext = new ResultExecutingContext(controllerContext, actionResult);
  5. Func<ResultExecutedContext> seed = delegate {
  6. this.InvokeActionResult(controllerContext, actionResult);
  7. return new ResultExecutedContext(controllerContext, actionResult, false, null);
  8. };
  9. return filters
          .Reverse<IResultFilter>()
          .Aggregate<IResultFilter, Func<ResultExecutedContext>>(seed,
            (next, filter) => () => InvokeActionResultFilter(filter, preContext, next))
          ();
  10. }

看这里方法的结构和上面Action那里差不多. 并且注意到这里的返回值:ResultExecutedContext, 返回的就是执行完View的结果.

先看一下 InvokeActionResultFilter方法

1. InvokeActionResultFilter

  1. internal static ResultExecutedContext InvokeActionResultFilter(IResultFilter filter,
       ResultExecutingContext preContext, Func<ResultExecutedContext> continuation)
  2. {
  3. filter.OnResultExecuting(preContext);
  4. if (preContext.Cancel)
  5. {
  6. return new ResultExecutedContext(preContext, preContext.Result, true, null);
  7. }
  8. bool flag = false;
  9. ResultExecutedContext filterContext = null;
  10. try
  11. {
  12. filterContext = continuation();
  13. }
  14. catch (ThreadAbortException)
  15. {
  16. filterContext = new ResultExecutedContext(preContext, preContext.Result, false, null);
  17. filter.OnResultExecuted(filterContext);
  18. throw;
  19. }
  20. catch (Exception exception)
  21. {
  22. flag = true;
  23. filterContext = new ResultExecutedContext(preContext, preContext.Result, false, exception);
  24. filter.OnResultExecuted(filterContext);
  25. if (!filterContext.ExceptionHandled)
  26. {
  27. throw;
  28. }
  29. }
  30. if (!flag)
  31. {
  32. filter.OnResultExecuted(filterContext);
  33. }
  34. return filterContext;
  35. }

内部方法的执行顺序都是一样的. 从这个方法能看出, Result 过滤器的执行顺序, 确实如前面提到的那样. 先执行Executing , 再执行 Result, 最后执行 Executed方法.

2. InvokeActionResult

  1. protected virtual void InvokeActionResult(ControllerContext controllerContext, ActionResult actionResult)
  2. {
  3. actionResult.ExecuteResult(controllerContext);
  4. }

这一步, 就是执行View了, 具体的执行过程有机会的话, 就在后面解析了.

这一篇主要是验证, Action/Result的四个过滤器的执行顺序是否如前面描述的那样子. 当然, 在探寻的过程中, 顺便也看到了Action/Result方法的执行时机.

这里其实还有一个问题, 从上面看, 把Controller排序到最前面执行了, 那么如果我是在FilterConfig中注册的呢? Action方法上标注的呢?Controller类上面标注的呢? 是一个什么顺序?

下一篇就去看一下, 这里的顺序, 也能辅助理解这一篇的内容.

参考:

  深入分析Asp.net Mvc

目录已同步

MVC源码分析 - Action/Result 过滤器执行时机的更多相关文章

  1. MVC源码分析 - Action/Result 过滤器(续)

    上一篇 看到了Action/Result过滤器的执行顺序: OnActionExecuting -> Action -> OnActionExecuted -> OnResultEx ...

  2. MVC源码分析 - Action查找和过滤器的执行时机

    接着上一篇, 在创建好Controller之后, 有一个 this.ExecuteCore()方法, 这部分是执行的. 那么里面具体做了些什么呢? //ControllerBaseprotected ...

  3. MVC源码分析 - Authorize授权过滤器

    从 上一篇 其实能看到, 程序执行的过滤器, 有四种 : 过滤器类型 接口 描述 Authorization IAuthorizationFilter 此类型(或过滤器)用于限制进入控制器或控制器的某 ...

  4. asp.net mvc源码分析-Action篇 IModelBinder

    我们首先还是看看ReflectedParameterBindingInfo的Binder属性吧: public override IModelBinder Binder {            ge ...

  5. Struts2 源码分析——DefaultActionInvocation类的执行action

    本章简言 上一章讲到关于拦截器的机制的知识点,让我们对拦截器有了一定的认识.我们也清楚的知道在执行用户action类实例之前,struts2会先去执行当前action类对应的拦截器.而关于在哪里执行a ...

  6. asp.net mvc源码分析-DefaultModelBinder 自定义的普通数据类型的绑定和验证

    原文:asp.net mvc源码分析-DefaultModelBinder 自定义的普通数据类型的绑定和验证 在前面的文章中我们曾经涉及到ControllerActionInvoker类GetPara ...

  7. ASP.NET MVC 源码分析(一)

    ASP.NET MVC 源码分析(一) 直接上图: 我们先来看Core的设计: 从项目结构来看,asp.net.mvc.core有以下目录: ActionConstraints:action限制相关 ...

  8. 精尽Spring MVC源码分析 - 寻找遗失的 web.xml

    该系列文档是本人在学习 Spring MVC 的源码过程中总结下来的,可能对读者不太友好,请结合我的源码注释 Spring MVC 源码分析 GitHub 地址 进行阅读 Spring 版本:5.2. ...

  9. 精尽Spring MVC源码分析 - HandlerAdapter 组件(三)之 HandlerMethodArgumentResolver

    该系列文档是本人在学习 Spring MVC 的源码过程中总结下来的,可能对读者不太友好,请结合我的源码注释 Spring MVC 源码分析 GitHub 地址 进行阅读 Spring 版本:5.2. ...

随机推荐

  1. _技巧_SublimeText_打开文件乱码解决

    macOS属于Unix分支,默认使用UTF-8编码,当从Window 或者其他Linux 或 Unix系统 拷贝文件过来,由于Window系统使用GBK或者GB2312中文编码,所以会出现乱码现象. ...

  2. index.jsp首页访问不了的解决方法

    解决方法: Tomcat,将项目添加到里面 部署解包的webapp目录 将Web项目部署到Tomcat中的方法之一,是部署没有封装到WAR文件中的Web项目.要使用这一方法部署未打包的webapp目录 ...

  3. 初始Hbase

    Hbase 定义 HBase是一个开源的非关系型分布式数据库(NoSQL),它参考了谷歌的BigTable建模,实现 的编程语言为 Java. 是Apache软件基金会的Hadoop项目的一部分,运行 ...

  4. POJ3046选蚂蚁创建集合_线性DP

    POJ3046选蚂蚁创建集合 一个人的精力是有限的呢,如果一直做一件事迟早会疲惫,所以自己要把握好,不要一直埋头于一件事,否则效率低下还浪费时间 题目大意:一共有T(1,2...n为其种类)种蚂蚁,A ...

  5. django创建分页

    前台html代码 <!DOCTYPE html> <html lang="en"> <head> <meta charset=" ...

  6. Java学习--基本数据类型的定义和运算2

    例1 public class OperateDemo01{ public static void main(String args[]){ int num = 22 ; System.out.pri ...

  7. cmd如何进入d盘

    首先打开CMD 点开始 运行输入 CMD 在CMD窗口中输入 CD\(就是返回根目录) 回车 在输入 D: 即可在D盘操作状态

  8. EBS-如何查看非自己提交的请求的结果

    http://www.cnblogs.com/quanweiru/p/4692071.html 如何查看非自己提交的请求的结果定位要找的请求SQL举例:SELECT req.request_id,   ...

  9. cxGrid实现取消过滤和排序后定位到首行(单选和多选)

    cxGrid实现取消过滤和排序后定位到首行(单选和多选) 原创 2013年10月06日 18:42:24 2107 DataContoller中的函数FocusedRecordIndex没有反应,Fo ...

  10. applicationContext.xml 基本配置

    <!-- 头文件,主要注意一下编码 --><?xml version="1.0" encoding="UTF-8"?><beans ...