[Web API] Web API 2 深入系列(4) Action的选择
目录
ApiController
HttpActionDescriptor
IHttpActionSelector
ApiController
在上节中,讲到如何选择并激活对应的IHttpController,而一般我们在开发中使用的是ApiController
public abstract class ApiController : IHttpController, IDisposable
{
public virtual Task<HttpResponseMessage> ExecuteAsync(HttpControllerContext controllerContext, CancellationToken cancellationToken)
{
this.Initialize(controllerContext);
HttpActionDescriptor actionDescriptor = services.GetActionSelector().SelectAction(controllerContext);
HttpActionContext actionContext = new HttpActionContext(controllerContext,actionDescriptor);
return services.GetActionInvoker().InvokeActionAsync(actionContext, cancellationToken);
}
}
在ApiController中,我们看到通过内置的DI容器选择出对应的HttpActionDescriptor.本节重点内容就是介绍SelectAction方法.
HttpActionDescriptor
在介绍IHttpActionSelector前,我们需要了解HttpActionDescriptor
public abstract class HttpActionDescriptor
{
//相关联的HttpControllerDescriptor
public HttpControllerDescriptor ControllerDescriptor { get; } { set; }
//Action名称(通过使用ActionName特性,Action名称可以和方法名称不同)
public abstract string ActionName { get; }
//Action返回值类型(void为null)
public abstract Type ReturnType { get; }
//支持的HttpMethod(默认一个Action方法支持一种HttpMethod,且默认为Post Method,可以使用AcceptVerbs特性支持多个)
public virtual Collection<HttpMethod> SupportedHttpMethods { get; }
//Action方法所有参数
public abstract Collection<HttpParameterDescriptor> Parameters { get; };
}
而HttpActionDescriptor默认实现为ReflectedHttpActionDescriptor,这里稍微展示下初始化HttpActionDescriptor过程
public class ReflectedHttpActionDescriptor : HttpActionDescriptor
{
//初始化HttpActionDescriptor的属性
public ReflectedHttpActionDescriptor(HttpControllerDescriptor controllerDescriptor, MethodInfo methodInfo)
: base(controllerDescriptor)
{
this.InitializeProperties(methodInfo);
this._parameters = new Lazy<Collection<HttpParameterDescriptor>>(() => this.InitializeParameterDescriptors());
}
private Collection<HttpParameterDescriptor> InitializeParameterDescriptors()
{
return this.ParameterInfos.Select(item => new ReflectedHttpParameterDescriptor((HttpActionDescriptor) this, item)).ToList();
}
}
IHttpActionSelector
IHttpActionSelector是选择Action最核心的类
public interface IHttpActionSelector
{
//选择符合标准的Action
HttpActionDescriptor SelectAction(HttpControllerContext controllerContext);
//获取HttpController所有Action
ILookup<string, HttpActionDescriptor> GetActionMapping(HttpControllerDescriptor controllerDescriptor);
}
IHttpActionSelector默认的实现为ApiControllerActionSelector
public class ApiControllerActionSelector : IHttpActionSelector
{
ILookup<string, HttpActionDescriptor> GetActionMapping(HttpControllerDescriptor controllerDescriptor);
{
//找到所有符合要求的方法
var methods = controllerDescriptor.ControllerType
.GetMethods(BindingFlags.Instance | BindingFlags.Public), methodInfo =>
!methodInfo.IsSpecialName && !methodInfo.GetBaseDefinition().DeclaringType.IsAssignableFrom(TypeHelper.ApiControllerType)
&& methodInfo.GetCustomAttribute<NonActionAttribute>() == null);
return methods.Select(method => new HttpActionDescriptor(controllerDescriptor,method))...;
}
}
注意:GetActionMapping会在请求每个HttpController第一次的时候 缓存当前所有HttpActionDescriptor
当我们找到HttpController下所有HttpActionDescriptor,还需要最后一步通过SelectAction筛选出最终的Action
由于这块源码稍微复杂,这里把关键的几步及对应的方法名说明下
public virtual HttpActionDescriptor SelectAction(HttpControllerContext controllerContext)
{
1. Action名称过滤
如果路由中配置Action变量,则会先过滤ActionName.调用方法GetInitialCandidateList
2. Http方法过滤
对于Get Put Post默认都通过,调用方法FindActionsForVerb
3. 必须的参数能绑定上
在Action方法上的必须参数,在路由参数和Request参数中能获取到,调用方法FindActionMatchRequiredRouteAndQueryParameters
4. 参数个数符合最多的匹配
当多个Action方法都满足以上3个条件时,取最多参数符合的那个,调用方法FindActionMatchMostRouteAndQueryParameters
5. 取唯一匹配
最终匹配结果为1个Action方法则成功,多个或零失败,在SelectAction方法本身调用
}
这里稍微举个例子来解释SelectAction
public class DemoController : ApiController
{
public string Get(int x)
public string Get(int x, int y)
public string Get(string x, string y)
public string Get(string x, string y, string z)
}
分别请求如下地址,并列出对应的匹配方法
demo?x=1 OK
public string Get(int x)
demo?x=1&y=2 Erro
public string Get(int x, int y)
public string Get(string x, string y)
demo?x=1&y=2&z=3 OK
public string Get(string x, string y, string z)
(OK表示请求成功,Erro表示请求失败)
备注:
- Action的选择明显比Controller的激活要复杂
- 文章中的代码并非完整WebAPI代码,一般是经过自己精简后的.
- 本篇内容使用MarkDown语法编辑
首发地址:http://neverc.cnblogs.com/p/5956432.html
[Web API] Web API 2 深入系列(4) Action的选择的更多相关文章
- ASP.NET Web API - ASP.NET MVC 4 系列
Web API 项目是 Windows 通信接口(Windows Communication Foundation,WCF)团队及其用户激情下的产物,他们想与 HTTP 深度整合.WCF ...
- 如何用Baas快速在腾讯云上开发小程序-系列1:搭建API & WEB WebSocket 服务器
版权声明:本文由贺嘉 原创文章,转载请注明出处: 文章原文链接:https://www.qcloud.com/community/article/221059001487422606 来源:腾云阁 h ...
- Java web与web gis学习笔记(二)——百度地图API调用
系列链接: Java web与web gis学习笔记(一)--Tomcat环境搭建 Java web与web gis学习笔记(二)--百度地图API调用 JavaWeb和WebGIS学习笔记(三)-- ...
- HTML5权威指南--Web Storage,本地数据库,本地缓存API,Web Sockets API,Geolocation API(简要学习笔记二)
1.Web Storage HTML5除了Canvas元素之外,还有一个非常重要的功能那就是客户端本地保存数据的Web Storage功能. 以前都是用cookies保存用户名等简单信息. 但是c ...
- 初试ASP.NET Web API/MVC API(附Demo)
写在前面 HTTP RESTful 创建Web API 调用Web API 运行截图及Demo下载 ASP.NET Web API是一个框架,可以很容易构建达成了广泛的HTTP服务客户端,包括浏览 ...
- 我所理解的RESTful Web API [Web标准篇]
REST不是一个标准,而是一种软件应用架构风格.基于SOAP的Web服务采用RPC架构,如果说RPC是一种面向操作的架构风格,而REST则是一种面向资源的架构风格.REST是目前业界更为推崇的构建新一 ...
- 重构Web Api程序(Api Controller和Entity)续篇
昨天有写总结<重构Web Api程序(Api Controller和Entity)>http://www.cnblogs.com/insus/p/4350111.html,把一些数据交换的 ...
- web api写api接口时返回
web api写api接口时默认返回的是把你的对象序列化后以XML形式返回,那么怎样才能让其返回为json呢,下面就介绍两种方法: 方法一:(改配置法) 找到Global.asax文件,在Applic ...
- Google Maps API Web Services
原文:Google Maps API Web Services 摘自:https://developers.google.com/maps/documentation/webservices/ Goo ...
随机推荐
- 基于AT89C51单片机的贪吃蛇电子游戏(仿真)
有关贪吃蛇的历史发展可以看一下这个网址,贪吃蛇最初的设计和现在并不相同..http://www.techweb.com.cn/internet/2013-02-21/1278055.shtml 该项目 ...
- .net core中使用openssl的公钥私钥进行加解密
这篇博文分享的是 C#中使用OpenSSL的公钥加密/私钥解密 一文中的解决方法在 .net core 中的改进.之前的博文针对的是 .NET Framework ,加解密用的是 RSACryptoS ...
- (翻译)《Hands-on Node.js》—— Introduction
今天开始会和大熊君{{bb}}一起着手翻译node的系列外文书籍,大熊负责翻译<Node.js IN ACTION>一书,而我暂时负责翻译这本<Hands-on Node.js> ...
- JS原型继承和类式继承
前言 一个多月前,卤煮读了一篇翻译过来的外国人写的技术博客.此君在博客中将js中的类(构造)继承和原型继承做了一些比较,并且得出了结论:建议诸位在开发是用原型继承.文中提到了各种原型继承的优点,详细的 ...
- Tomcat搭建
标签:Tomcat 概述 Tomcat 服务器是一个免费的开放源代码的Web 应用服务器,属于轻量级应用服务器,在中小型系统和并发访问用户不是很多的场合下被普遍使用,是开发和调试JSP 程序的首选. ...
- Atom使用纪要
一直在更新的原文地址奉上,欢迎PR:Atom使用纪要 官网地址: atom.io 目前(2015/7/29)Atom主题已有725个:Package已有2394 简单交代背景 Atom 是 Githu ...
- 使用WCF的Trace与Message Log功能
原创地址:http://www.cnblogs.com/jfzhu/p/4030008.html 转载请注明出处 前面介绍过如何创建一个WCF Service http://www.cnblo ...
- 老司机学新平台 - Xamarin Forms开发框架二探 (Prism vs MvvmCross)
在上一篇Xamarin开发环境及开发框架初探中,曾简单提到MvvmCross这个Xamarin下的开发框架.最近又评估了一些别的,发现老牌Mvvm框架Prism现在也支持Xamarin Forms了, ...
- wpf ListView DataTemplate方式的鼠标悬停和选中更改背景色
今天使用wpf技术弄一个ListView的时候,由于需求需要,需要ListView显示不同的数据模板,很自然的使用了DataTemplate方式来定义多个数据模板,并在ListView中使用ItemT ...
- 每天一个linux命令(47):iostat命令
Linux系统中的 iostat是I/O statistics(输入/输出统计)的缩写,iostat工具将对系统的磁盘操作活动进行监视.它的特点是汇报磁盘活动统计情况,同时也会汇报出CPU使用情况. ...