MVC源码分析 - Action/Result 过滤器执行时机
前面 的篇章, 解析了Action方法的查找, 以及 Authorize, Action, Result, Error 过滤器的加载时机. 也花了两篇去看授权和错误过滤器的使用. 但是对于 Action/Result 的执行以及Action/Result里面的两个过滤器的执行时机, 并没有清晰看到.
一、Action 方法的执行和过滤器的执行
//System.Web.Mvc.ControllerActionInvoker
protected virtual ActionExecutedContext InvokeActionMethodWithFilters(ControllerContext controllerContext,
IList<IActionFilter> filters, ActionDescriptor actionDescriptor, IDictionary<string, object> parameters)
{
ActionExecutingContext preContext = new ActionExecutingContext(controllerContext, actionDescriptor, parameters);
//这里定义了一个Func委托, 返回值为 ActionExecutedContext 类型
//在返回的 ActionExecutedContext 类型实例中, 对 Result 指定了执行方法
//这里其实都是在指定待执行的方法, 在它执行完毕后, 并没有任何的方法被执行. 以下标红的部分, 就是执行 Action 方法
Func<ActionExecutedContext> seed = () => new ActionExecutedContext(controllerContext, actionDescriptor, false, null)
{ Result = this.InvokeActionMethod(controllerContext, actionDescriptor, parameters) };
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))
(); //这一步执行方法
}
这里的代码就是 前面 的 3.4, 只是我对他进行了分割, 这样容易看懂点
这里有一个 Aggregate<>() 可能不是那么好理解, 先上一个demo, 应该是能够促进理解的.
var numbers = new int[] { , , , };
// 使用当前总值乘以下一个数值得到新的总值
var product = numbers.Aggregate((total, next) => total * next);
最后的值是: 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
internal static ActionExecutedContext InvokeActionMethodFilter(IActionFilter filter,
ActionExecutingContext preContext, Func<ActionExecutedContext> continuation)
{
//这一步执行 ActionExecuting 过滤器
filter.OnActionExecuting(preContext);
if (preContext.Result != null)
{
//当不能通过过滤器的时候, 会走这里
return new ActionExecutedContext(preContext, preContext.ActionDescriptor, true, null)
{ Result = preContext.Result };
}
bool flag = false;
ActionExecutedContext filterContext = null;
try
{
//1.第一次执行这一步,会跳转到 InvokeActionFilter(filter, preContext, next)中去执行
//2.第二次运行到这里的时候,会去执行 上面的 seed 方法
//在执行seed方法的时候, 就回去运行 InvokeActionMethod() 方法
filterContext = continuation();
}
catch (ThreadAbortException)
{
filterContext = new ActionExecutedContext(preContext, preContext.ActionDescriptor, false, null);
filter.OnActionExecuted(filterContext);
throw;
}
catch (Exception exception)
{
flag = true;
filterContext = new ActionExecutedContext(preContext, preContext.ActionDescriptor, false, exception);
filter.OnActionExecuted(filterContext);
if (!filterContext.ExceptionHandled)
{
throw;
}
}
if (!flag)
{
filter.OnActionExecuted(filterContext);
}
return filterContext;
}
在这里, 已经能看到 ActionExecuting 和 ActionExecuted 两个过滤器的执行时机了
根据上面的注释, 然后看一下 InvokeActionMethod 方法
protected virtual ActionResult InvokeActionMethod(ControllerContext controllerContext,
ActionDescriptor actionDescriptor, IDictionary<string, object> parameters)
{
object actionReturnValue = actionDescriptor.Execute(controllerContext, parameters);
return this.CreateActionResult(controllerContext, actionDescriptor, actionReturnValue);
}
这里的 Execute 方法, 返回的就是 ActionResult了, 调用CreateActionResult 是为了进行类型转换, 将返回的actionReturnValue 转换为ActionResult类型.
这里其实就是执行的 Action 方法. 并返回执行结果.
二、Result 的执行和过滤器的执行
//ControllerActionInvoker
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))
();
}
看这里方法的结构和上面Action那里差不多. 并且注意到这里的返回值:ResultExecutedContext, 返回的就是执行完View的结果.
先看一下 InvokeActionResultFilter方法
1. InvokeActionResultFilter
internal static ResultExecutedContext InvokeActionResultFilter(IResultFilter filter,
ResultExecutingContext preContext, Func<ResultExecutedContext> continuation)
{
filter.OnResultExecuting(preContext);
if (preContext.Cancel)
{
return new ResultExecutedContext(preContext, preContext.Result, true, null);
}
bool flag = false;
ResultExecutedContext filterContext = null;
try
{
filterContext = continuation();
}
catch (ThreadAbortException)
{
filterContext = new ResultExecutedContext(preContext, preContext.Result, false, null);
filter.OnResultExecuted(filterContext);
throw;
}
catch (Exception exception)
{
flag = true;
filterContext = new ResultExecutedContext(preContext, preContext.Result, false, exception);
filter.OnResultExecuted(filterContext);
if (!filterContext.ExceptionHandled)
{
throw;
}
}
if (!flag)
{
filter.OnResultExecuted(filterContext);
}
return filterContext;
}
内部方法的执行顺序都是一样的. 从这个方法能看出, Result 过滤器的执行顺序, 确实如前面提到的那样. 先执行Executing , 再执行 Result, 最后执行 Executed方法.
2. InvokeActionResult
protected virtual void InvokeActionResult(ControllerContext controllerContext, ActionResult actionResult)
{
actionResult.ExecuteResult(controllerContext);
}
这一步, 就是执行View了, 具体的执行过程有机会的话, 就在后面解析了.
这一篇主要是验证, Action/Result的四个过滤器的执行顺序是否如前面描述的那样子. 当然, 在探寻的过程中, 顺便也看到了Action/Result方法的执行时机.
这里其实还有一个问题, 从上面看, 把Controller排序到最前面执行了, 那么如果我是在FilterConfig中注册的呢? Action方法上标注的呢?Controller类上面标注的呢? 是一个什么顺序?
下一篇就去看一下, 这里的顺序, 也能辅助理解这一篇的内容.
参考:
MVC源码分析 - Action/Result 过滤器执行时机的更多相关文章
- MVC源码分析 - Action/Result 过滤器(续)
上一篇 看到了Action/Result过滤器的执行顺序: OnActionExecuting -> Action -> OnActionExecuted -> OnResultEx ...
- MVC源码分析 - Action查找和过滤器的执行时机
接着上一篇, 在创建好Controller之后, 有一个 this.ExecuteCore()方法, 这部分是执行的. 那么里面具体做了些什么呢? //ControllerBaseprotected ...
- MVC源码分析 - Authorize授权过滤器
从 上一篇 其实能看到, 程序执行的过滤器, 有四种 : 过滤器类型 接口 描述 Authorization IAuthorizationFilter 此类型(或过滤器)用于限制进入控制器或控制器的某 ...
- asp.net mvc源码分析-Action篇 IModelBinder
我们首先还是看看ReflectedParameterBindingInfo的Binder属性吧: public override IModelBinder Binder { ge ...
- Struts2 源码分析——DefaultActionInvocation类的执行action
本章简言 上一章讲到关于拦截器的机制的知识点,让我们对拦截器有了一定的认识.我们也清楚的知道在执行用户action类实例之前,struts2会先去执行当前action类对应的拦截器.而关于在哪里执行a ...
- asp.net mvc源码分析-DefaultModelBinder 自定义的普通数据类型的绑定和验证
原文:asp.net mvc源码分析-DefaultModelBinder 自定义的普通数据类型的绑定和验证 在前面的文章中我们曾经涉及到ControllerActionInvoker类GetPara ...
- ASP.NET MVC 源码分析(一)
ASP.NET MVC 源码分析(一) 直接上图: 我们先来看Core的设计: 从项目结构来看,asp.net.mvc.core有以下目录: ActionConstraints:action限制相关 ...
- 精尽Spring MVC源码分析 - 寻找遗失的 web.xml
该系列文档是本人在学习 Spring MVC 的源码过程中总结下来的,可能对读者不太友好,请结合我的源码注释 Spring MVC 源码分析 GitHub 地址 进行阅读 Spring 版本:5.2. ...
- 精尽Spring MVC源码分析 - HandlerAdapter 组件(三)之 HandlerMethodArgumentResolver
该系列文档是本人在学习 Spring MVC 的源码过程中总结下来的,可能对读者不太友好,请结合我的源码注释 Spring MVC 源码分析 GitHub 地址 进行阅读 Spring 版本:5.2. ...
随机推荐
- ssh+注解开发 pom.xml
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/20 ...
- toast js
参考别人的,自己改写了下,很好用. <!DOCTYPE html> <html lang="zh-CN"> <head> <meta ch ...
- docker mac
官网上有安装方法.已安装成功. 容器千万不能随便删 现在已经不用boot2docker,使用dockertools来安装docker,dockertools中包含docker-machine,等等. ...
- AngularJS 单元测试 Karma jasmine
当AngularJS项目越来越大时候,需要进行单元测试,可以先开发功能再进行测试,也可以先进行测试. 一.karma 是一个基于Node.js(先要安装)的JavaScript测试执行过程管理工具( ...
- 软件测试实践平台(Mooctest)FAQ
0. 背景 我们在<软件测试技术>课程使用 MOOCTEST (mooctest.net) 作为课程的实践教学平台. 在教学过程中学生们遇到了一些问题,本文将一一总结. 注意:本文在解决问 ...
- adb错误 - INSTALL_FAILED_NO_MATCHING_ABIS
#背景 换组啦,去了UC国际浏览器,被拥抱变化了.还在熟悉阶段,尝试了下adb,然后就碰到了这个INSTALL_FAILED_NO_MATCHING_ABIS的坑... #解决方法 INSTALL_F ...
- 绿色版Mysql自动建立my.ini和命令行启动并动态指定datadir路径
1.先去下载绿色版的Mysql(https://cdn.mysql.com//archives/mysql-5.7/mysql-5.7.20-winx64.zip) 2.解压缩到任意目录(如D:\My ...
- nodejs学习(imooc课程笔记, 主讲人Scott)
课程地址: 进击Node.js基础(一) 进击Node.js基础(二) 1. nodejs创建服务器 var http = require('http'); //加载http模块 //请求进来时, 告 ...
- 全球第一开源ERP Odoo操作手册 数据库简介
1.3 数据库简介 每一个独立核算的企业都有一套相互关联的账簿体系, 把这一套完整的账簿体系建立在计算机系统中就称为一个数据库. 一般一个企业只用一个数据库. 如果企业有几个下属的独立核算的实体,也可 ...
- 【BZOJ2882】 工艺(SAM)
传送门 BZOJCH 洛谷 Solution 这个东西要求的不就是最小表示法吗? 把原串复制一遍然后都加到后缀自动机里面去. 用个map跑一下,这样子可以保证每一次选的是最小字典序的. 然后跑\(n\ ...