Asp.net web Api源码分析-HttpParameterBinding

接着上文Asp.net web Api源码分析-Filter 我们提到filter的获取和调用,后面通过HttpActionBinding actionBinding = actionDescriptor.ActionBinding;来获取HttpActionBinding实例,然后调用 HttpActionBinding的ExecuteBindingAsync方法来绑定Action参数。HttpActionDescriptor中 定义了ActionBinding属性,默认的实现代码:

 ServicesContainer controllerServices = _controllerDescriptor.Configuration.Services;
 IActionValueBinder actionValueBinder = controllerServices.GetActionValueBinder();
 HttpActionBinding actionBinding = actionValueBinder.GetBinding(this);

这里的actionValueBinder默认就是一个DefaultActionValueBinder实例,然后调用它的GetBinding方法。然我们看看DefaultActionValueBinder的GetBinding方法:

  public virtual HttpActionBinding GetBinding(HttpActionDescriptor actionDescriptor)
        {
            if (actionDescriptor == null)
            {
                throw Error.ArgumentNull("actionDescriptor");
            }

            HttpParameterDescriptor[] parameters = actionDescriptor.GetParameters().ToArray();
            HttpParameterBinding[] binders = Array.ConvertAll(parameters, GetParameterBinding);

            HttpActionBinding actionBinding = new HttpActionBinding(actionDescriptor, binders);

            EnsureOneBodyParameter(actionBinding);


            return actionBinding;
        }

这里首先获取当前HttpActionDescriptor的参数集合ReflectedHttpParameterDescriptor[],然后依次调用GetParameterBinding方法把当前HttpActionDescriptor转化为HttpParameterBinding,GetParameterBinding方法如下:

  protected virtual HttpParameterBinding GetParameterBinding(HttpParameterDescriptor parameter){// Attribute has the highest precedence// Presence of a model binder attribute overrides.ParameterBindingAttribute attr = parameter.ParameterBinderAttribute;if (attr != null){return attr.GetBinding(parameter);}// No attribute, so lookup in global map.ParameterBindingRulesCollection pb = parameter.Configuration.ParameterBindingRules;if (pb != null){HttpParameterBinding binding = pb.LookupBinding(parameter);if (binding != null){return binding;}}// Not explicitly specified in global map or attribute.// Use a default policy to determine it. These are catch-all policies. Type type = parameter.ParameterType;if (TypeHelper.IsSimpleUnderlyingType(type) || TypeHelper.HasStringConverter(type)){// For simple types, the default is to look in URI. Exactly as if the parameter had a [FromUri] attribute.return parameter.BindWithAttribute(new FromUriAttribute());}// Fallback. Must be a complex type. Default is to look in body. Exactly as if this type had a [FromBody] attribute.attr = new FromBodyAttribute();return attr.GetBinding(parameter);}

 这里我们首先调用 ParameterBindingAttribute attr = parameter.ParameterBinderAttribute;来获取ParameterBindingAttribute实例,然后调用它的 GetBinding方法来获取HttpParameterBinding,我们还是来看看ParameterBinderAttribute是如何定义 了吧:

在HttpParameterDescriptor中的ParameterBinderAttribute属性定义如下:

 public virtual ParameterBindingAttribute ParameterBinderAttribute
        {
            get
            {
                if (_parameterBindingAttribute == null)
                {
                    if (!_searchedModelBinderAttribute)
                    {
                        _searchedModelBinderAttribute = true;
                        _parameterBindingAttribute = FindParameterBindingAttribute();
                    }

                }

                return _parameterBindingAttribute;
            }

            set { _parameterBindingAttribute = value; }
        }

FindParameterBindingAttribute其实就是查找参数或则参数类型的ParameterBindingAttribute特性,如果这里取到的ParameterBindingAttribute属性多余一个则最后要抛出异常,具体这里就不说了。

回到DefaultActionValueBinder中GetParameterBinding方法,找不到 ParameterBindingAttribute对象,我们就 parameter.Configuration.ParameterBindingRules来获取 ParameterBindingRulesCollection实例,其中 ParameterBindingRules=DefaultActionValueBinder.GetDefaultParameterBinders(), 其中GetDefaultParameterBinders方法实现如下:

 internal static ParameterBindingRulesCollection GetDefaultParameterBinders()
        {
            ParameterBindingRulesCollection pb = new ParameterBindingRulesCollection();

            pb.Add(typeof(CancellationToken), parameter => new CancellationTokenParameterBinding(parameter));
            pb.Add(typeof(HttpRequestMessage), parameter => new HttpRequestParameterBinding(parameter));

            // Warning binder for HttpContent.
            pb.Add(parameter => typeof(HttpContent).IsAssignableFrom(parameter.ParameterType) ?
                                    parameter.BindAsError(Error.Format(SRResources.ParameterBindingIllegalType, parameter.ParameterType.Name, parameter.ParameterName))
                                    : null);

            return pb;
        }

这里的ParameterBindingRulesCollection实例有3个成员,然后调用 ParameterBindingRulesCollection的LookupBinding方法来获取HttpParameterBinding实 例,其中LookupBinding方法如下:

public class ParameterBindingRulesCollection : Collection<Func<HttpParameterDescriptor, HttpParameterBinding>>
{
     private static Func<HttpParameterDescriptor, HttpParameterBinding> TypeCheck(Type type, Func<HttpParameterDescriptor, HttpParameterBinding> func)
    {
        return (param => (param.ParameterType == type) ? func(param) : null);
    }
    public void Add(Type typeMatch, Func<HttpParameterDescriptor, HttpParameterBinding> funcInner)
    {
        Add(TypeCheck(typeMatch, funcInner));
    }
    public HttpParameterBinding LookupBinding(HttpParameterDescriptor parameter)
    {
        foreach (Func<HttpParameterDescriptor, HttpParameterBinding> func in this)
        {
            HttpParameterBinding binding = func(parameter);

            if (binding != null)
            {
                 return binding;
            }
        }
        return null;
    }
}

所以这里默认的ParameterBindingRulesCollection3个成员是不会返回HttpParameterBinding实例。

如果参数类型是一个简单类型,并且该类型可以转化为string类型,然后调用  return parameter.BindWithAttribute(new FromUriAttribute());返回HttpParameterBinding,BindWithAttribute方法其实就是调用ParameterBindingAttributeGetBinding方法,这里默认FromUriAttribute的GetBinding方法,这里FromUriAttribute的继承数如下:FromUriAttribute-》ModelBinderAttribute-》ParameterBindingAttribute。这里的

如果绑定的参数数据类型比较特殊,那么这里我们就调用FromBodyAttribute的GetBinding方法来获取HttpParameterBinding实例,这里的FromBodyAttribute继承树如下:

FromBodyAttribute-》ParameterBindingAttribute

在这里我们总结一下这里找HttpParameterBinding的顺序,(1)parameter.ParameterBinderAttribute实际就是找参数或参数类型的ParameterBindingAttribute属性,(2)

 parameter.Configuration.ParameterBindingRules 从全局的ParameterBindingRules中找HttpParameterBinding,(3)如果参数类型是一个简单类型且可以转化为 string那么我们调用parameter.BindWithAttribute(new FromUriAttribute()),(4)最后我们调用FromBodyAttribute的GetBinding方法来获取 HttpParameterBinding实例

现在我们回到DefaultActionValueBinder的GetBinding方法中来,现在我们已经获取到HttpParameterBinding集合,接下里创建一个HttpActionBinding实例,最后调用EnsureOneBodyParameter来检查HttpActionBinding的ParameterBindings集合是否有2个都需要读取form表单,如果是则抛出异常。到这里HttpActionDescriptor的ActionBinding的创建也就很清楚了。

回到ApiController的ExecuteAsync方法中来,这里继续调用HttpActionBinding的ExecuteBindingAsync方法,这里的ExecuteBindingAsync方法实现如下:

  public virtual Task ExecuteBindingAsync(HttpActionContext actionContext, CancellationToken cancellationToken)
        {
            if (_parameterBindings.Length == 0)
            {
                return TaskHelpers.Completed();
            }


            // First, make sure the actionBinding is valid before trying to execute it. This keeps us in a known state in case of errors.
            foreach (HttpParameterBinding parameterBinder in ParameterBindings)
            {
                if (!parameterBinder.IsValid)
                {
                    // Throwing an exception because the webService developer's action signature is bad.
                    // This exception will be caught and converted into a 500 by the dispatcher
                    return TaskHelpers.FromError(new InvalidOperationException(parameterBinder.ErrorMessage));
                }
            }

            if (_metadataProvider == null)
            {
                HttpConfiguration config = actionContext.ControllerContext.Configuration;
                _metadataProvider = config.Services.GetModelMetadataProvider();
            }

            // Execute all the binders.
            IEnumerable<Task> tasks = from parameterBinder in ParameterBindings select parameterBinder.ExecuteBindingAsync(_metadataProvider, actionContext, cancellationToken);
            return TaskHelpers.Iterate(tasks, cancellationToken, disposeEnumerator: false);
        }

这个方法逻辑很简单,如果ParameterBindings没有成员则直接返回,如果有则需要依次验证他们的IsValid,然后再依次调用他们的ExecuteBindingAsync方法,ExecuteBindingAsync方法的具体执行我们这里就不多说了。这里我们看看metadataProvider 是个什么东东吧,     SetSingle<ModelMetadataProvider>(new DataAnnotationsModelMetadataProvider());这里我们就知道metadataProvider 其实是一个DataAnnotationsModelMetadataProvider实例,其构造函数也没什么特别的。这里返回的Task的具体实现我就不多说了,里面用到了一个TaskCompletionSource类,具体的使用我这里一而不说了。

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/403355.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

TCP/IP详解卷1中文版勘误表前言

相信仔细阅读过TCP/IP这一经典著作中文版的读者们最痛苦的就是其中一些语句或者词汇感觉无法理解&#xff0c;其后果要么是无法理解&#xff0c;要么理解错误&#xff0c;如果错误的概念在脑中根深蒂固了&#xff0c;对于以后的学习和工作将是十分令人苦恼的事情。看到多数读者…

promise使用详解

原文链接以及promise练习题 先说结论&#xff1a; promise是解决异步函数的一种方案 将异步操作以同步方式展现出来 避免了回调地狱 1.三种状态&#xff1a;padding–等待 resolved–成功–then rejected–失败–catch 2.promise上有then和catch方法 then接受一个参数是函数 这…

vim 使用中的一些错误[omnifunc未设置错误]

From: http://blog.sina.com.cn/s/blog_60c70b6c01015b43.html 1.option ‘omnifunc’ is notset 错误: vim7下Omnicompletion默认情况下是没有开启的&#xff0c;有时候自定义的vimrc文件会实现自动补齐&#xff0c;例如vim-autocomplpop等等&#xff0c;在编辑html/css文件的…

Sqlite 管理工具收藏

1.SQLite Administrator http://download.orbmu2k.de/files/sqliteadmin.zip 2.SQLite2009Pro-v3.8.3.1 http://osenxpsuite.net/SQLite2009Pro-v3.8.3.1.zip 3.SqliteDev http://www.sqlitedeveloper.com/downloads/SqliteDev450.zip 4.sqlite_maestro_executable已破解.rar…

Android 对话框(Dialog)大全 建立你自己的对话框

Activities提供了一种方便管理的创建、保存、回复的对话框机制&#xff0c;例如 onCreateDialog(int), onPrepareDialog(int, Dialog), showDialog(int), dismissDialog(int)等方法&#xff0c;如果使用这些方法的话&#xff0c;Activity将通过getOwnerActivity()方法返回该Act…

SAP使用表空间传输异构系统迁移

SAP使用表空间传输异构系统迁移 此文档乃原创总结&#xff0c;仅适用于oracle数据库。此方法特点&#xff0c;速度快、停机时间短&#xff0c;配合RMAN使用甚至可以做到不停机、在线迁移并可以支持裸设备和ASM。以次方法做迁移&#xff0c;在实际测试中&#xff0c;从Solaris s…

在MAC下安装Exuberant ctags

From: http://blog.csdn.net/dragon1225/article/details/7021064 最简单的方式就是在终端执行&#xff1a;brew install ctags MAC自带的ctags文件只支持几种语言, 因为最近需要开发android程序, 使用到java语言, 所以下个Exuberant Ctags安装上~ 从http://ctags.sourceforge…

Eclipse: select at least one project

在Eclipse导入工程的时候如果遇到这种错误&#xff0c;一般是由于当前导入操作的工程名与Eclipse中已经导入的工程名重复所致。此时需要打开当前工程列表&#xff0c;删除重名目录即可&#xff1a;    PS: Window -> Show View -> Project Explorer 打开工程列表。

HDU 5795

题意&#xff1a; 两人轮流从 n 堆 糖果里取糖果&#xff0c;每次可以1&#xff09; 选择一堆取任意个&#xff08;不为 0&#xff09;2) 选择一堆糖果分成 3 堆&#xff08;每堆数量 > 1)拿到最后一颗糖果的人赢。 解题 &#xff1a; 打表算出一些 sg 值&#xff0c;就可以…

IOS UIPageController

- (void)viewDidLoad{[super viewDidLoad];// Do any additional setup after loading the view, typically from a nib.//定义UIScrollView//self.scrollView [[UIScrollView alloc] init];//self.scrollView.frame CGRectMake(10, 0, 200, 500);self.scrollView.contentSiz…

mac下beyond compare配置图

配置Tab键为4个空格&#xff1a; 显示行号&#xff1a;

子网寻址

一、概念   将IP地址中的主机号再分成一个子网号和主机号。而不是把IP地址看成单纯的网络号主机号。举例来说&#xff1a;一个B类网络地址&#xff0c;网络位为16位&#xff0c;主机位为16位。若划分子网则需要对这16主机位继续进行划分&#xff0c;比如在这16bit中将8bit用…

Win7下安装linux虚拟机

关于如何在Win7下搭建linux学习环境&#xff0c;特在此分享下. 一、工具 1、VMware-workstation-full-9.0.0-812388.exe 下载地址:http://pan.baidu.com/s/1bpFT0ZX 2、32位的CenOS系统镜像文件CentOS-6.7-i386-bin-DVD1.iso. 下载地址:http://pan.baidu.com/s/1cHrxLG…

el-table处理某一行样式;el-table表格row-class-name无效原因;el-table格式化某行样式和数据

如果是设置点击el-table某行变色或者hover 点击看这篇 情景&#xff1a;:row-class-name“tableRowClassName” 无效果 原因是&#xff1a;el-table的属性里 多了个stripe&#xff0c;stripe属性可以创建带斑马纹的表格。会影响到row-class-name的渲染。去掉stripe即可。 &l…

国内ios分亨组件,

2019独角兽企业重金招聘Python工程师标准>>> 1. 新浪微博 (支持新浪微博官方 SSO OAuth 及 iOS 6 特性&#xff09; 2. 微信&#xff08;仅支持分享到朋友&#xff09; 3. 网易微博 4. 腾讯微博 5. 豆瓣 6. 人人网 https://github.com/icyl…

子网掩码

子网掩码用于确定网络地址中子网号的位数&#xff08;也即比特数&#xff09;及主机号位数。掩码是一个32bit的值&#xff0c;其中值为1的比特留给网络号和子网号&#xff0c;为0的比特留给主机号。譬如一个B类地址&#xff0c;子网掩码可以设置为&#xff1a;   <1> …

让linux的SSH客户端也能记住服务器的密码

From: http://blog.chinaunix.net/uid-17240700-id-2813900.html CU上一位网友的问题如下: 以前一直用windows&#xff0c;最近工作原因换Ubuntu了&#xff0c;但是windows下的SSH客户端(我用的是Netsarang公司的 Xshell,Xftp,Xmanager)&#xff0c;linux没有相应版本…

el-table文字超出隐藏;el-table表格文字一行展示,超出隐藏,多余的内容会在 hover时显示 ;

el-table文字默认情况下若内容过多会折行显示&#xff0c;若需要单行显示可以给el-table-column使用show-overflow-tooltip属性&#xff0c;它接受一个Boolean&#xff0c;为true时多余的内容会在 hover 时以 tooltip 的形式显示出来。 给el-table-column添加show-overflow-t…

webdriverAPI-Java

1.1 下载selenium2.0的lib包 http://code.google.com/p/selenium/downloads/list 官方UserGuide&#xff1a;http://seleniumhq.org/docs/ 1.2 用webdriver打开一个浏览器 我们常用的浏览器有firefox和IE两种&#xff0c;firefox是selenium支持得比较成熟的浏览器。但是做页…