[Web API] Web API 2 深入系列(5) 特性路由
目录
1. 特性路由注册
2. 路由解析
- 生成DataTokens
- 选择HttpController
- 选择Action
特性路由的目的在于更好的提供restful架构的接口,最近好忙(懒),所以更新速度慢.
特性路由注册
[Route(模板)] :定义特性路由模板
- 普通变量
a/b/{c}
- 缺省变量
a/b/{c=d}
- 变量约束
a/b/{c:int:range(10,20)}
- 通配符
a/b/{*c:datetime}
[RoutePrefix("api/demo")] :定义路由前缀
路由解析
通过IRoutePrefix/IHttpRouteInfoProvider,我们可以直接注册路由,映射到具体的Controller和Action.
当调用MapHttpAttributeRoutes方法时,WebAPI会创建1个唯一的RouteCollectionRoute作为IHttpRoute并添加到路由表中.
MapHttpAttributeRoutes方法:
public static void MapHttpAttributeRoutes(HttpConfiguration configuration, IInlineConstraintResolver constraintResolver, IDirectRouteProvider directRouteProvider)
{
RouteCollectionRoute aggregateRoute = new RouteCollectionRoute();
configuration.Routes.Add("MS_attributerouteWebApi", (IHttpRoute) aggregateRoute);
Action<HttpConfiguration> previousInitializer = configuration.Initializer;
configuration.Initializer = (Action<HttpConfiguration>) (config =>
{
previousInitializer(config);
aggregateRoute.EnsureInitialized((Func<IReadOnlyCollection<IHttpRoute>>) (() =>
{
subRoutes = new SubRouteCollection();
AttributeRoutingMapper.AddRouteEntries(subRoutes, configuration, constraintResolver, directRouteProvider);
return subRoutes;
}));
});
}
RouteCollectionRoute是特性路由的HttpRoute对象,既是一个IHttpRoute对象,又是一个IHttpRoute集合.并且其中核心方法为GetRouteData(IHttpRoute其他接口都返回为null),
internal class RouteCollectionRoute : IHttpRoute, IReadOnlyCollection<IHttpRoute>, IEnumerable<IHttpRoute>, IEnumerable
{
public IHttpRouteData GetRouteData(string virtualPathRoot, HttpRequestMessage request)
{
List<IHttpRouteData> httpRouteDataList = new List<IHttpRouteData>();
//调用内部的SubRoutes对象
foreach (IHttpRoute subRoute in (IEnumerable<IHttpRoute>) this.SubRoutes)
{
IHttpRouteData routeData = subRoute.GetRouteData(virtualPathRoot, request);
httpRouteDataList.Add(routeData);
}
return (IHttpRouteData) new RouteCollectionRoute.RouteCollectionRouteData((IHttpRoute) this, httpRouteDataList.ToArray());
}
}
在该方法中,我们发现RouteCollectionRoute调用了内部所有的SubRoutes对象.
而其内部的SubRoutes类型实际为SubRouteCollection类型
internal class SubRouteCollection : IReadOnlyCollection<IHttpRoute>, IEnumerable<IHttpRoute>, IEnumerable
{
private readonly List<IHttpRoute> _routes = new List<IHttpRoute>();
private readonly List<RouteEntry> _entries = new List<RouteEntry>();
public IReadOnlyCollection<RouteEntry> Entries{get;}
}
而SubRoutes的创建是在MapHttpAttributeRoutes方法定义,实际调用是在HttpServer的Send方法初始化的.
protected override async Task<HttpResponseMessage> SendAsync(HttpRequestMessage request, CancellationToken cancellationToken)
{
this.EnsureInitialized();
//...
}
首先验证一下我们的特性路由注册位置(定义1个扩展方法)
public static class RouteCollectionExt
{
public static IEnumerable<IHttpRoute> GetSubRoutes(this HttpRouteCollection routes)
{
var route = routes["MS_attributerouteWebApi"];
var prop = route.GetType().GetProperty("SubRoutes", BindingFlags.Instance | BindingFlags.NonPublic);
var subRoutes = prop.GetValue(route) as IEnumerable<IHttpRoute>;
return subRoutes;
}
}
生成DataTokens
DataTokens这次发挥了一定的作用,同时也告诉我们该如何使用它. (在第1节中,我觉得DataTokens是个冗余设计)
先看下DataTokens上有哪些东西.
private static void ShowSubRoutesTokens(HttpRouteCollection routes)
{
foreach (var subRoute in routes.GetSubRoutes())
{
Console.WriteLine(subRoute.RouteTemplate);
foreach (var dataToken in subRoute.DataTokens)
{
Console.WriteLine("{0,-12}{1}", dataToken.Key, dataToken.Value);
}
Console.WriteLine();
}
}
截图:

对于DataTokens的actions和precedence的属性,在DirectRouteBuilder的Build方法中实现
其中actions表示该路由模板对应的actiondescription(在特性路由中,会为每个controller创建独立的一份子路由.)
而precedence表示匹配的优先级,对于有约束的优先级高于无优先级.(约束分为常量,变量,通配符)
在前2节中,我们讲了如何选择Action以及Controller.
实际上,如果使用特性路由.选择的机制又有些变化.
选择HttpController
public virtual HttpControllerDescriptor SelectController(HttpRequestMessage request)
{
IHttpRouteData routeData = request.GetRouteData();
if (routeData != null)
{
//在GetDirectRouteController内获取了特性路由对应的Controller,同时要求匹配的所有特性路由对应的Controller为同一个
HttpControllerDescriptor directRouteController = DefaultHttpControllerSelector.GetDirectRouteController(routeData);
return directRouteController;
}
//普通路由方式
string controllerName = this.GetControllerName(request);
//...
}
选择Action
public HttpActionDescriptor SelectAction(HttpControllerContext controllerContext)
{
var matchingActions = this.FindMatchingActions(controllerContext, false);
//...
}
private List<ApiControllerActionSelector.CandidateActionWithParams> FindMatchingActions(HttpControllerContext controllerContext, bool ignoreVerbs = false)
{
//此处做特性路由判断
IEnumerable<IHttpRouteData> subRoutes = controllerContext.RouteData.GetSubRoutes();
return subRoutes == null ? 普通路由 : 特性路由;
}
备注:
- 如果我们为特性路由指定了Name,则会自动创建一个IHttpRoute绑定到RouteCollection上.(这步是在HttpConfiguration初始化中最后做判断完成的)
- 文章中的代码并非完整WebAPI代码,一般是经过自己精简后的.
- 本篇内容使用MarkDown语法编辑
首发地址:http://neverc.cnblogs.com/p/5975086.html
[Web API] Web API 2 深入系列(5) 特性路由的更多相关文章
- ASP.NET WEB API必知必会:特性路由
一.什么是特性路由? 特性路由是指将RouteAttribute或自定义继承自RouteAttribute的特性类标记在控制器或ACTION上,同时指定路由Url字符串,从而实现路由映射,相比之前的通 ...
- ASP.NET WEB API 特性路由
一.什么是特性路由? 特性路由是指将RouteAttribute或自定义继承自RouteAttribute的特性类标记在控制器或ACTION上,同时指定路由Url字符串,从而实现路由映射,相比之前的通 ...
- [Web API] Web API 2 深入系列(6) Model绑定(上)
目录 解决什么问题 Model元数据解析 复杂类型 ValueProvider ValueProviderFactory 解决什么问题 Model: Action方法上的参数 Model绑定: 对Ac ...
- 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 ...
- 我所理解的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,把一些数据交换的 ...
随机推荐
- 使用Javascript来实现二级联动菜单的效果
效果图如下: 具体实现步骤如下: 1.所用js代码如下: <script type="text/javascript"> var arr_province=[" ...
- Android操作HTTP实现与服务器通信(转)
Android操作HTTP实现与服务器通信 本示例以Servlet为例,演示Android与Servlet的通信. 众所周知,Android与服务器通信通常采用HTTP通信方式和Socket通信方 ...
- TCP/IP, WebSocket 和 MQTT
按照OSI网络分层模型,IP是网络层协议,TCP是传输层协议,而HTTP和MQTT是应用层的协议.在这三者之间, TCP是HTTP和MQTT底层的协议.大家对HTTP很熟悉,这里简要介绍下MQTT.M ...
- 采访Philipp Crocoll:安卓平台上整合Java和C#
在这个采访中,我们跟开源开发者Philipp Crocoll讨论了关于Keepass2Android的相关话题.Keepass2Android不仅具有强大的密码存储的功能,还是在一个单独的安卓应用同时 ...
- Hadoop学习笔记—9.Partitioner与自定义Partitioner
一.初步探索Partitioner 1.1 再次回顾Map阶段五大步骤 在第四篇博文<初识MapReduce>中,我们认识了MapReduce的八大步凑,其中在Map阶段总共五个步骤,如下 ...
- NodeJS写个爬虫,把文章放到kindle中阅读
这两天看了好几篇不错的文章,有的时候想把好的文章 down 下来放到 kindle 上看,便写了个爬虫脚本,因为最近都在搞 node,所以就很自然的选择 node 来爬咯- 本文地址:http://w ...
- Step by step Install a Local Report Server and Remote Report Server Database
原创地址:http://www.cnblogs.com/jfzhu/p/4012097.html 转载请注明出处 前面的文章<Step by step SQL Server 2012的安装 &g ...
- 注册OCX失败
今天注册某个OCX时,Windows报告以下错误: 模块“XXX.ocx”已加载,但对 DllRegisterServer 的调用失败,错误代码为 0x80040200. 这是Windows权限引起的 ...
- WCF:传输EntityFramework 实体类的POCO 代理
WCF传输EntityFramework 实体类的POCO 代理 Windows Communication Foundation (WCF) 不能对代理进行直接序列化或反序列化,因为 DataCon ...
- C语言 · 图形显示
问题描述 编写一个程序,首先输入一个整数,例如5,然后在屏幕上显示如下的图形(5表示行数): * * * * * * * * * * * * * * * #include "stdi ...