ASP.NET Web API Model-ActionBinding
ASP.NET Web API Model-ActionBinding
前言
前面的几个篇幅把Model部分的知识点划分成一个个的模块来讲解,而在控制器执行过程中分为好多个过程,对于控制器执行过程(一)主要讲解了过滤器以及在后面的过滤器篇幅中也有讲到,而在过滤器之中还有一些执行过程,也就是在授权过滤器执行完毕后,行为过滤器执行之前,我们要做的就是Model绑定,面前也都说了之前对Model的知识点模块都讲解的差不多了,今天这个篇幅我们就来看一下这些零散知识点的源头,也就是Model绑定的入口点。
Model-ActionBinding
HttpActionBinding的由来
我们通过前面几篇的了解都知道在ASP.NET Web API框架中进入整个Model绑定的入口点就是在HttpActionBinding类型中,对于这个类型前面的篇幅也介绍过,它里面封装了ParameterBinding数组,这些ParameterBinding就是控制器方法中每个参数执行Model绑定的对象了,既然我们知道HttpActionBinding类型中有着许多ParameterBinding类型的对象实例,那么我们就要看看HttpActionBinding类型是怎么生成的。
示例代码1-1
this.SetSingle<IActionValueBinder>(new DefaultActionValueBinder());
首先我们看到示例代码1-1中可以看到在HttpConfiguration类型的服务容器中,默认注册为IActionValueBinder类型服务的是DefaultActionValueBinder类型。
示例代码1-2
namespace System.Web.Http.ModelBinding
{
public class DefaultActionValueBinder : IActionValueBinder
{
public DefaultActionValueBinder(); public virtual HttpActionBinding GetBinding(HttpActionDescriptor actionDescriptor);
protected virtual HttpParameterBinding GetParameterBinding(HttpParameterDescriptor parameter);
}
}
代码1-2中所示的是DefaultActionValueBinder类型的定义,其中的这两个方法很重要,第一个GetBinding()方法是用以框架内部来进行调用,根据HttpActionDescriptor控制器方法描述类型对象获取到我们所需的HttpActionBinding,而其内部实现则是调用下面的GetParameterBinding()方法,利用HttpActionDescriptor对象获取到HttpParameterDescriptor集合后,然后遍历的去调用GetParameterBinding()方法,从而能够获取到HttpParameterBinding对象实例,最后生成HttpActionBinding对象实例,从设计角度来看这个DefaultActionValueBinder类型中的两个方法GetBinding()和GetParameterBinding()方法都是采用了template method模式,这种模式在框架设计中很常见。
HttpParameterBinding的由来
下面我们就要来说说GetParameterBinding()方法的细节实现了因为关乎着使用哪种方式来进行绑定。也就是根据HttpParameterDescriptor类型实例怎么去创建HttpParameterBinding的。
示例代码1-3
protected virtual HttpParameterBinding GetParameterBinding(HttpParameterDescriptor parameter)
{
ParameterBindingAttribute parameterBinderAttribute = parameter.ParameterBinderAttribute;
if (parameterBinderAttribute == null)
{
ParameterBindingRulesCollection parameterBindingRules = parameter.Configuration.ParameterBindingRules;
if (parameterBindingRules != null)
{
HttpParameterBinding binding = parameterBindingRules.LookupBinding(parameter);
if (binding != null)
{
return binding;
}
}
Type parameterType = parameter.ParameterType;
if (TypeHelper.IsSimpleUnderlyingType(parameterType) || TypeHelper.HasStringConverter(parameterType))
{
return parameter.BindWithAttribute(new FromUriAttribute());
}
parameterBinderAttribute = new FromBodyAttribute();
}
return parameterBinderAttribute.GetBinding(parameter);
}
代码1-3就是具体的实现了,那我们就就来看一下其中的过程以及会涉及到的类型。
首先会根据参数HttpParameterDescriptor类型实例获取到在这个控制器方法参数上使用了ParameterBindingAttribute标识,并且获取ParameterBindingAttribute类型实例。我们暂且就来看一下ParameterBindingAttribute类型定义。
示例代码1-4
[AttributeUsage(AttributeTargets.Parameter | AttributeTargets.Class, Inherited = true, AllowMultiple = false)]
public abstract class ParameterBindingAttribute : Attribute
{
// Methods
protected ParameterBindingAttribute();
public abstract HttpParameterBinding GetBinding(HttpParameterDescriptor parameter);
}
从代码1-4可以看出,这个ParameterBindingAttribute类型适用于类型以及参数,也就是说我们选择绑定的方式可以在Model类型定义的时候标识这个特性,也可以在定义控制器方法的时候适当的给参数这个特性标识。
然而在这个类型中为什么会有GetBinding()方法呢?因为这个类型是抽象类型,也就是采用了上面所说过的template method模式,而在子类实现中,根据自身适应的情况生成响应的HttpParameterBinding类型。看下如下的图表示相关的对象模型。
图1

上面图1中涉及到的每个类型大家可以去看前面的篇幅,篇幅中有遗漏的就麻烦大家自己多动一下手去看看吧。
HttpParameterBinding的选择机制
接着代码1-3的思绪,在我们获取到了ParameterBindingAttribute之后,并不知道这个控制器方法中的参数是否标识有ParameterBindingAttribute,或者是参数类型上是否有标识。这个时候假使是有的话,可以看到代码1-3中的最后一句代码,直接使用获取到的ParameterBindingAttribute类型进行调用GetBinding()方法,也就是在上一小节中图1所示的那样。
然而还有一种情况,就是我们在定义控制器方法的时候参数没有明确的标识我们要使用某种绑定机制,或者是在定义Model的时候没有明确的表示,这个时候框架则会从定义好的规则集合中根据当前控制其方法参数的描述类型来获取对应的ParameterBinding类型实例。如下的示例代码定义了规则集合的定义。
示例代码1-5
internal static ParameterBindingRulesCollection GetDefaultParameterBinders()
{
ParameterBindingRulesCollection ruless = new ParameterBindingRulesCollection();
ruless.Add(typeof(CancellationToken), parameter => new CancellationTokenParameterBinding(parameter));
ruless.Add(typeof(HttpRequestMessage), parameter => new HttpRequestParameterBinding(parameter));
ruless.Add(delegate (HttpParameterDescriptor parameter) {
if (!typeof(HttpContent).IsAssignableFrom(parameter.ParameterType))
{
return null;
}
return parameter.BindAsError(Error.Format(SRResources.ParameterBindingIllegalType, new object[] { parameter.ParameterType.Name, parameter.ParameterName }));
});
return ruless;
}
代码1-5中所表示的就是规则定义,意思就是在我们使用HttpParameterDescriptor类型实例来从集合中想获取ParameterBinding的时候,ParameterBindingRulesCollection类型会把我们的HttpParameterDescriptor类型实例中的ParameterType取出来和之前定义的每一项规则的类型进行比对,类型吻合了就会随之调用对应的委托类型进行ParameterBinding生成。从代码1-5中我们可以看到的是规则中只有CancellationToken类型和HttpRequestMessage类型,假使这个时候我们的控制其方法参数类型是自定义的复杂类型,这里也都没有定义,这个时候框架会取出HttpParameterDescriptor类型中的ParameterType进行判断,假使是可以转换成string类型的简单类型参数,则会生成一个FromUriAttribute类型作为标识,FromUriAttribute类型继承自ModelBinderAttribute类型。
假使这里的判断没有通过则说明是复杂类型,最后我们再看待代码1-3中的定义最后生成的是FromBodyAttribute标识类型,这个时候请参照图1。
作者:金源
出处:http://www.cnblogs.com/jin-yuan/
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面
ASP.NET Web API Model-ActionBinding的更多相关文章
- Model Validation in ASP.NET Web API
Model Validation in ASP.NET Web API 原文:http://www.asp.net/web-api/overview/formats-and-model-binding ...
- Asp.Net Web API 2第十五课——Model Validation(模型验证)
前言 阅读本文之前,您也可以到Asp.Net Web API 2 系列导航进行查看 http://www.cnblogs.com/aehyok/p/3446289.html 本文参考链接文章地址htt ...
- ASP.NET Web API Model-ParameterBinding
ASP.NET Web API Model-ParameterBinding 前言 通过上个篇幅的学习了解Model绑定的基础知识,然而在ASP.NET Web API中Model绑定功能模块并不是被 ...
- ASP.NET Web API 过滤器创建、执行过程(二)
ASP.NET Web API 过滤器创建.执行过程(二) 前言 前面一篇中讲解了过滤器执行之前的创建,通过实现IFilterProvider注册到当前的HttpConfiguration里的服务容器 ...
- 在一个空ASP.NET Web项目上创建一个ASP.NET Web API 2.0应用
由于ASP.NET Web API具有与ASP.NET MVC类似的编程方式,再加上目前市面上专门介绍ASP.NET Web API 的书籍少之又少(我们看到的相关内容往往是某本介绍ASP.NET M ...
- ASP.NET Web API Model-ModelBinder
ASP.NET Web API Model-ModelBinder 前言 本篇中会为大家介绍在ASP.NET Web API中ModelBinder的绑定原理以及涉及到的一些对象模型,还有简单的Mod ...
- ASP.NET Web API Model-ValueProvider
ASP.NET Web API Model-ValueProvider 前言 前面一篇讲解了Model元数据,Model元数据是在Model绑定中很重要的一部分,只是Model绑定中涉及的知识点比较多 ...
- ASP.NET Web API Model-ModelMetadata
ASP.NET Web API Model-ModelMetadata 前言 前面的几个篇幅主要围绕控制器的执行过程,奈何执行过程中包含的知识点太庞大了,只能一部分一部分的去讲解,在上两篇中我们看到在 ...
- ASP.NET Web API 控制器执行过程(一)
ASP.NET Web API 控制器执行过程(一) 前言 前面两篇讲解了控制器的创建过程,只是从框架源码的角度去简单的了解,在控制器创建过后所执行的过程也是尤为重要的,本篇就来简单的说明一下控制器在 ...
随机推荐
- Spark踩坑记——Spark Streaming+Kafka
[TOC] 前言 在WeTest舆情项目中,需要对每天千万级的游戏评论信息进行词频统计,在生产者一端,我们将数据按照每天的拉取时间存入了Kafka当中,而在消费者一端,我们利用了spark strea ...
- .NET Core & ASP.NET Core 1.0在Redhat峰会上正式发布
众所周知,Red Hat和微软正在努力使.NET Core成为Red Hat企业版Linux (RHEL)系统上的一流开发平台选项.这个团队已经一起工作好几个月了,RHEL对.NET有许多需求.今天在 ...
- java基础_集合List与Set接口
List接口继承了Collection的方法 当然也有自己特有的方法向指定位置添加元素 add(索引,添加的元素); 移除指定索引的元素 remove(索引) 修改指定索引的元素 set ...
- 使用技术手段限制DBA的危险操作—Oracle Database Vault
概述 众所周知,在业务高峰期,某些针对Oracle数据库的操作具有很高的风险,比如修改表结构.修改实例参数等等,如果没有充分评估和了解这些操作所带来的影响,这些操作很可能会导致故障,轻则导致应用错误, ...
- 深入研究Visual studio 2017 RC新特性
在[Xamarin+Prism开发详解三:Visual studio 2017 RC初体验]中分享了Visual studio 2017RC的大致情况,同时也发现大家对新的Visual Studio很 ...
- ASP.NET MVC5----常见的数据注解和验证
只要一直走,慢点又何妨. 在使用MVC模式进行开发时,数据注解是经常使用的(模型之上操作),下面是我看书整理的一些常见的用法. 什么是验证,数据注解 验证 从全局来看,发现逻辑仅是整个验证的很小的一部 ...
- 验证管理员权限(C#)
参考页面: http://www.yuanjiaocheng.net/webapi/test-webapi.html http://www.yuanjiaocheng.net/webapi/web-a ...
- 基于注解的bean配置
基于注解的bean配置,主要是进行applicationContext.xml配置.DAO层类注解.Service层类注解. 1.在applicationContext.xml文件中配置信息如下 &l ...
- stringstream的基本用法
原帖地址:https://zhidao.baidu.com/question/580048330.htmlstringstream是字符串流.它将流与存储在内存中的string对象绑定起来.在多种数据 ...
- 多线程同步工具——Lock
本文原创,转载请注明出处. 参考文章: <"JUC锁"03之 公平锁(一)> <"JUC锁"03之 公平锁(二)> 锁分独占锁与共享锁, ...