ASP.NET Web API中的action参数类型可以分为简单类型和复杂类型。

HttpResponseMessage Put(int id, Product item)

id是int类型,是简单类型,item是Product类型,是复杂类型。

简单类型实参值从哪里读取呢?
--一般从URI中读取

所谓的简单类型包括哪些呢?
--int, bool, double, TimeSpan, DateTime, Guid, decimal, string,以及能从字符串转换而来的类型

复杂类型实参值从哪里读取呢?
--一般从请求的body中读取

复杂类型实参值是否可以从URI中获取呢?
--可以,按如下

→ 有这样的一个类

public class Shape
{
public double Width{get;set;}
public double Length{get;set;}
}

→ 想从URI中获取,那就加上[FromUri]

public HttpResponseMessage Get([FromUri] Shape shape)

→ 客户端就可以放在查询字符串中传

...api/products/?Width=88&Length=199

简单类型可以从请求的body中获取吗?
--可以。按如下:

→ action方法

public HttpResponseMessage Post([FromBody] string name){...}

→ 前端请求中

Content-Type:applicaiton/json

"hello world"

API服务端会根据Content-Type的值选择合适的媒体类型。

复杂类型是否可以从uri中的字符串获取呢?
--可以

api/products/?shape=188,80

如何把uri中查询字符串中shape的字段值,即以逗号分割的字符串转换成Shape类实例呢?
--使用TypeConverter类

[TypeConverter(typeof(ShapeConverter))]
public class Shape
{
public double Width{get;set;}
public double Length{get;set;} public static bool TryParse(string s, out Shampe result)
{
result = null;
var parts = s.Split(',');
if(parts.lenth != )
{
return false;
} double width, length; if(double.TryParse(parts[], out width) && double.TryParse(parts[], out length))
{
result = new Shape(){Width = width; Length = length};
return true;
}
return false;
}
} public class ShapeConverter: TypeConverter
{
public override bool CanConvertFrom(ITypeDescriptorContext context, Type sourcType)
{
if(sourceType == typeof(string))
{
return true;
} return base.CanConvertFrom(context, sourceType);
} public override object ConvertFrom(ITypeDescriptorContext context, CultureInfo, object value)
{
if(value is string)
{
Shape shape;
if(Shape.TryParse((string)value, out shape))
{
return shape;
}
}
return base.ConvertFrom(context, culture, value);
}
}

→ 在action不需要[FromUri]

public HttpResponseMessage Get(Shape shape)

→ 客户端

api/products/?shape=188,80

是否可以通过Model Binder来实现自定义参数绑定过程呢?
--可以,有IModelBinder接口,提供了BindModel方法

→ 自定义一个Model Binder

public class ShapeModelBinder : IModelBinder
{
private static ConcurrentDictionary<string, Shape> _shapes = new ConcurrentDictionary<string, Shape>(StringComparer.OrdinalIgnoreCase); static ShapeModelBinder()
{
_shapes["shape1"] = new Shape(){Width= , Length = };
_shapes["shape2"] = new Shape(){Width=, Length = };
} public bool BindModel(HttpActionContext actionContext, ModelBindingContect biningContext)
{
if(bindingContext.ModelType != typeof(Shape))
{
return false;
} ValueProviderResult val = bindingContext.ValueProvider.GetValue(bidingContext.ModelName);
if(val == null)
{
return false;
} string key = val.RawValue as string;
if(key == null){
bdingContext.ModelState.AddModelError(bindingContext.ModelName, "值类型错误");
return false;
} Shape shape;
if(_shapes.TryGetValue(key, out shape) || Shape.TryParse(key, shape))
{
bindingContext.Model = result;
return true;
} bindingContext.ModelState.AddModelError(bindingContext.ModelName, "无法把字符串转换成Shape");
return false;
}
}

● 从BindingContext中的ValueProvider属性获取到ValueProviderResult
● 从前端查询字符串中传来的字符串,被放在ValueProviderResult的RawValue属性中
● 把字符串转换成Shape实例,最终放在了BindingContext的Model属性中

→ 使用自定义的Model Binder

可以运用在action中:

public HttpResposneMessage Get([ModelBinder(typeof(ShapeModelBinder))] Shape shape);

可以放在模型上:

[ModelBinder(typeof(Shape))]
public class Shape
{ }

也可以放在全局注册中:

public static class WebApiConfig
{
public static void Register(HttpConfiguraiton config)
{
var provider = new SimpleModelBinderProvider(typeof(Shape), new ShapeModelBinder());
config.Services.Insert(typeof(ModelBinderProvider), , provider);
}
}

注意:即使在全局注册,也需要在action中按如下写:

public HttpResponseMessage Get([ModelBinder] Shape shape);

是否可以通过Value Provider来自定义参数绑定过程呢?
--可以。

比如,从前端cookie中获取值,自定义一个Value Provider.

public class MyCookieValueProvider : IValueProvider
{
private Dictionary<string, string> _values; public MyCookieValueProvider(HttpActionContext actionContext)
{
if(actionContext == null)
{
throw new ArgumentNullException("actionContext");
} _values = new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase);
foreach(var cookie in actionContext.Request.Headers.GetCookies())
{
foreach(CookieState state in cookie.Cookies)
{
_values[state.Name] = state.Value;
}
}
} public bool COntainsPrefix(string prefix)
{
return _values.keys.Contains(prefix);
} public ValueProviderResult GetValue(string key)
{
string value;
if(_values.TryGetValue(key, out value))
{
return new ValueProviderResult(value, value, CultureInfo.InvariantCulture);
}
return null;
}
}

同时还需要一个ValueProviderFactory.

public class MyCookieValueProviderFactory : ValueProviderFactory
{
public override IValueProvider GetValueProvider(HttpActionContext actionContext)
{
return new MyCookeValueProvider(actionContext);
}
}

最后注册到全局中。

public static void Register(HttpConfiguration config)
{
config.Services.Add(typeof(ValueProviderFactory), new MyCookieValueProviderFactory());
}

还可以把自定义的ValueProvider放在action中。

public HttpResponseMessage Get([ValueProvider(typeof(MyCookieValueProviderFactory))] Shape shape);

是否可以通过HttpParameterBinding实现参数绑定自定义呢?
--可以。

ModelBinderAttribute继承于ParameterBindingAttribute.

public abstract class ParameterBindingAttribute : Attribute
{
public abstract HttpParameterBinding GetBinding(HttpParameterDescriptor parameter);
}

HttpParameterBinding用来把值绑定到参数上。

假设,需要从前端请求的if-match和if-none-match字段获取ETag值。

public class ETag
{
public string Tag{get;set;}
}

可能从if-match获取,也可能从if-none-match获取,来个枚举。

public enum ETagMatch
{
IfMatch,
IfNoneMatch
}

自定义HttpParameterBinding。

public class ETagParameterBinding : HttpParameterBinding
{
ETagMatch _match; public ETagParameterBinding(HttpParameterDescriptor parameter, ETagMatch match) : base(parameter)
{
_match = match;
} public override Task ExecuteBindingAsync(ModelMetadataProvider metadataProvider, HttpActionContext actionContext, CancellationToken candellationToken)
{
EntityTagHeaderValue etagHeader = null;
switch(_match)
{
case ETagMatch.IfNoneMatch:
etagHeader = actionContext.Request.Headers.IfNoneMatch.FirstOrDefault();
break;
case ETagMatch.IfMatch:
etagHeader = actionContext.Request.Headers.IfMatch.FirstOrDefault();
break;
} ETag etag = null;
if(etagHeader != null)
{
etag = new ETag{Tag = etagHeader.Tag};
} actionContext.ActionArguemnts[Descriptor.ParameterName] = etag; var tsc = new TaskCompletionSource<object>();
tsc.SetResult(null);
return tsc.Task;
}
}

可见,所有的action参数放在了ActionContext的ActionArguments中的。

如何使用自定义的HttpParameterBinding呢?
--通过自定义一个ParameterBindingAttribute特性。

public abstract class ETagMatchAttribute : ParameterBindingAttribute
{
private ETagMatch _match; public ETagMatchAttribute(ETagMatch match)
{
_match = match;
} public override HttpParameterBinding GetBinding(HttpParameterDescriptor parameter)
{
if(parameter.ParameterType == typeof(ETag))
{
return new ETagParameterBinding(parameter, _match);
}
return parameter.BindAsError("参数类型不匹配");
}
} public class IfMatchAttribute : ETageMatchAttribute
{
public IfMatchAttribute(): base(ETagMatch.IfMatch)
{}
} public class IfNoneMatchAttribute: ETagMatchAttribute
{
public IfNoneMatchAttribute() : base(ETagMatch.IfNoneMatch)
{}
}

再把定义的有关HttpParameterBinding的特性运用到方法上。

public HttpResponseMessage Get([IfNoneMatch] ETag etag)

还需要在全局注册:

config.ParameterBindingRules.Add(p => {
if(p.ParameterType == typeof(ETag) && p.ActionDescriptor.SupportedHttpMethods.Contains(HttpMethod.Get))
{
return new ETagParameterBinding(p, ETagMatch.IfNoneMatch);
}
else
{
return null;
}
})

总结,本篇体验了简单类型和复杂类型获取前端数据的方式。并通过自定义ValueProvider, ModelBinder, HttpParameterBinding来实现对参数绑定过程的控制。

ASP.NET Web API中的参数绑定总结的更多相关文章

  1. ASP.NET Web API中的Controller

    虽然通过Visual Studio向导在ASP.NET Web API项目中创建的 Controller类型默认派生与抽象类型ApiController,但是ASP.NET Web API框架本身只要 ...

  2. ASP.NET Web API中的JSON和XML序列化

    ASP.NET Web API中的JSON和XML序列化 前言 阅读本文之前,您也可以到Asp.Net Web API 2 系列导航进行查看 http://www.cnblogs.com/aehyok ...

  3. 利用查询条件对象,在Asp.net Web API中实现对业务数据的分页查询处理

    在Asp.net Web API中,对业务数据的分页查询处理是一个非常常见的接口,我们需要在查询条件对象中,定义好相应业务的查询参数,排序信息,请求记录数和每页大小信息等内容,根据这些查询信息,我们在 ...

  4. 【ASP.NET Web API教程】6.2 ASP.NET Web API中的JSON和XML序列化

    谨以此文感谢关注此系列文章的园友!前段时间本以为此系列文章已没多少人关注,而不打算继续下去了.因为文章贴出来之后,看的人似乎不多,也很少有人对这些文章发表评论,而且几乎无人给予“推荐”.但前几天有人询 ...

  5. Asp.Net Web API 2第十三课——ASP.NET Web API中的JSON和XML序列化

    前言 阅读本文之前,您也可以到Asp.Net Web API 2 系列导航进行查看 http://www.cnblogs.com/aehyok/p/3446289.html 本文描述ASP.NET W ...

  6. 【ASP.NET Web API教程】4.3 ASP.NET Web API中的异常处理

    原文:[ASP.NET Web API教程]4.3 ASP.NET Web API中的异常处理 注:本文是[ASP.NET Web API系列教程]的一部分,如果您是第一次看本系列教程,请先看前面的内 ...

  7. 【ASP.NET Web API教程】4.1 ASP.NET Web API中的路由

    原文:[ASP.NET Web API教程]4.1 ASP.NET Web API中的路由 注:本文是[ASP.NET Web API系列教程]的一部分,如果您是第一次看本博客文章,请先看前面的内容. ...

  8. 目标HttpController在ASP.NET Web API中是如何被激活的:目标HttpController的创建

    目标HttpController在ASP.NET Web API中是如何被激活的:目标HttpController的创建 通过上面的介绍我们知道利用HttpControllerSelector可以根据 ...

  9. ASP.NET Web API中的Routing(路由)

    [译]Routing in ASP.NET Web API 单击此处查看原文 本文阐述了ASP.NET Web API是如何将HTTP requests路由到controllers的. 如果你对ASP ...

随机推荐

  1. topsort | | jzoj[1226] | | NOIP2003神经网络

    今天终于通过了那道永远都看不懂题目的神经网络... 所谓拓扑排序,就是在有向无环图中,根据已经有的点和点之间的关系进行排序 引用jzyz教材上的栗子:比如说奶牛比较食量大小,我现在拿到的是cow[i] ...

  2. AT91-PWM应用

    步骤1: make menuconfig配置内核, 开启PWM输出功能. Device Drivers ---> Misc devices  ---> <*>Atmel AT3 ...

  3. CAST 类型转换应用

    1: select 2: ID,SystemID,Department, 3: case Number when 0 then '若干' else CAST(Number as varchar)+'人 ...

  4. 快速创建node应用[Express框架]

    安装Express npm install -g express 建立工程 express -e ejs FaceExpresscd FaceExpress && npm instal ...

  5. C++混合编程之idlcpp教程Python篇(3)

    上一篇 C++混合编程之idlcpp教程Python篇(2) 是一个 hello world 的例子,仅仅涉及了静态函数的调用.这一篇会有新的内容. 与PythonTutorial0相似,工程Pyth ...

  6. [算法]——全排列(Permutation)以及next_permutation

    排列(Arrangement),简单讲是从N个不同元素中取出M个,按照一定顺序排成一列,通常用A(M,N)表示.当M=N时,称为全排列(Permutation).从数学角度讲,全排列的个数A(N,N) ...

  7. VC中LINK 2001 和 LINK 2009 的错误的解决

    最近将两个开源C++项目编译成windows版本的时候遇到很多问题,关键是两个项目经过同事的修改之后,一个项目引用了另一个项目,两个项目的头文件中都有一些跨平台的关于数据类型,以及一些通用函数的定义, ...

  8. protobuf-net 与 C#中几种序列化的比较

    C#中几种序列化的比较,此次比较只是比较了 序列化的耗时和序列后文件的大小. 几种序列化分别是: 1. XmlSerializer 2. BinaryFormatter 3. DataContract ...

  9. JS更随机的随机数

    一.问题背景 一个二维平面上有一群NPC,每一回合可以随机向上/下/左/右任一方向走1步,有单位碰撞体积(NPC位置不能重合) 规则就这么简单,初始情况下这群NPC是被人工均匀分布在二维平面上的,运行 ...

  10. CoreCLR中超过3万行代码的gc.cpp文件的来源

    在CoreCLR的开源代码中,GC的主要实现代码gc.cpp文件大小竟然有1.17MB,打开文件一看,竟然有35490行!第一次见到如此多行的单个代码文件. github都不让直接查看:https:/ ...