[水煮 ASP.NET Web API2 方法论](12-2)管理 OD…
2018-06-22 06:06:05来源:未知 阅读 ()
问题
如何控制 OData 路由
解决方案
为了注册路由,可以使用 HttpConfigurationExtension 类中 MapODataServiceRoute 的扩展方法。对于单一路由这样做足以,其余的处理由实体数据模型来处理。
config.MapODataServiceRoute("OData", "OData", builder.GetEdmModel());
从 ASP.NET Web API 2.2 开始支持 OData 直接声明路由,在 Action 上使用 ODataRouteAttribute。这和常规的属性路由一样,可以通过 ODataRoutePrefixAttribute 在 Controller 级别设置路由前缀.
1 [ODataRoute("Players")] 2 3 public IQueryable<Player> GetAllPlayers() 4 5 { 6 7 // 忽略 8 9 }
工作原理
OData 在 Web API 中路由是通过 ODataRoute 类实现的,其实,他是 HttpRoute 的一个子类。定制路由需要支持 ODataPathRouteConstraint,他是 OData 制定的,是IHttpRouteConstrain 的实现类,是为了确保所有的 OData 属性或 OData 路由约定在路由匹配后能够设置到 HttpRequestMessage 上。
ODataPath 类是用来包装转换 OData 资源路径,并以强类型的方式公开段(segment)。按约定,ASP.NET Web API 使用资源路径(URI 的一部分,例如,/Player),他是基于实体数据模型来映射到相应的 Controller。然后,动词依据动作选择在 Controller 中找到相应的 Action。此外,在 Web API 中 OData 可以使用自定义路由,这些路由都会影响动作选择。
小提示 更多关于 OData 路由约定的信息,请戳这里
https://www.asp.net/web-api/overview/OData-support-in-aspnet-web-api/OData-routing-conventions
我们可以通过我们自己的 IODataRoutingConvertion 重写 OData 路由行为、实现自定义 OData Controller、实现动作选择,如清单 12-3 所示
清单 12-3 IODataRoutingConvention 定义
1 public interface IODataRoutingConvention 2 3 { 4 5 string SelectController(ODataPath ODataPath, HttpRequestMessage request); 6 7 string SelectAction(ODataPath ODataPath, HttpControllerContext controllerContext, 8 9 ILookup<string, HttpActionDescriptor> actionMap); 10 11 }
当我们定义 OData 路由时,可以通过 MapODataServiceRoue 扩展方法来自定义路由约定;其中个的一个重载方法就是 使用 IODataRoutingConvention 的集合作为参数的方法。如果没有传值(例如,在使用 MapODataServiceRoute),那么,Web API 将在内部调用静态的 ODataRoutingConventions。CreateDefaultWithAttributeRouting 只使用默认的内建路由约定。
在 OData 中属性路由是 IDataRoutingConvention 的另一个版本-AttributeRoutingCinvention。他会查找所有 ODataRouteAttribute 的用法,使用 Dictionary<ODataPathTemplate,HttpActionDescriptot> 的形式建立适当的映射关系。如果进来的请求与当前 HTTP 请求的 ODataPath 匹配,那么,与此相关的 HttpActionDescriptor Controller 将会被选翻牌子来处理请求。
属性路由对于非标准路由来说是不错的选择,例如,使用非绑定的 OData 功能或 Action 的时候。尝试使用集中路由的方式进行路由,就需要自定义路由约定,然而,属性路由可以使用 ODataRouteAttribute 相关方法,用很简单的声明方式直接完成。
需要注意的是属性路由与常规路由不同,他是默认启用的。也就是说,除非我们重写默认的 IODataRoutingConventions,否则,Web API 会调用 ODataRoutingConvents。无论什么时候使用 MapODataServiceRoute 方法,CreateDefaultWithAttributeRouting 内部都会确保被 Web API OData 使用的 AttributeRoutingConvention 包含在约定的集合中。如清单 12-4 所示,摘录自 Web API 源码。
清单 12-4 ODataRoutingConvrention 类,确保 AttributeRoutingConvention 被包含。
1 public static class ODataRoutingConventions 2 3 { 4 5 public static IList<IODataRoutingConvention> CreateDefaultWithAttributeRouting( 6 7 HttpConfiguration configuration, 8 9 IEdmModel model) 10 11 { 12 13 if (configuration == null) 14 15 { 16 17 throw Error.ArgumentNull("configuration"); 18 19 } 20 21 if (model == null) 22 23 { 24 25 throw Error.ArgumentNull("model"); 26 27 } 28 29 IList<IODataRoutingConvention> routingConventions = CreateDefault(); 30 31 AttributeRoutingConvention routingConvention = new AttributeRoutingConvention(model, 32 33 configuration); 34 35 routingConventions.Insert(0, routingConvention); 36 37 return routingConventions; 38 39 } 40 41 public static IList<IODataRoutingConvention> CreateDefault() 42 43 { 44 45 return new List<IODataRoutingConvention>() 46 47 { 48 49 new MetadataRoutingConvention(), 50 51 new EntitySetRoutingConvention(), 52 53 new SingletonRoutingConvention(), 54 55 new EntityRoutingConvention(), 56 57 new NavigationRoutingConvention(), 58 59 new PropertyRoutingConvention(), 60 61 new RefRoutingConvention(), 62 63 new ActionRoutingConvention(), 64 65 new FunctionRoutingConvention(), 66 67 new UnmappedRequestRoutingConvention() 68 69 }; 70 71 } 72 73 }
结果就是,在应用程序启动,不在需要调用任何其他的方法通知框架扫描所有的路由属性。事实上,只有这样,才能在最开始的地方获取属性路由并调用 MapODataServiceRoute.
代码演示
为了介绍 OData Web API 的集中路由,我们只需要在 HttpConfiguration 中调用 MapODataServiceRoute 和传路由前缀,以及我们的 IEdmModel。一个完整的启动类列子,使用简单的 OData 实体和集中路由,如清单 12-5 所示,使用 Player 实体。
清单 12-5. 声明一个基本的 OData 路由启动类
1 public class Startup 2 3 { 4 5 public void Configuration(IAppBuilder builder) 6 7 { 8 9 var ODataBuilder = new ODataConventionModelBuilder(); 10 11 ODataBuilder.EntitySet<Player>("Players"); 12 13 var edm = ODataBuilder.GetEdmModel(); 14 15 var config = new HttpConfiguration(); 16 17 config.MapODataServiceRoute("Default OData", "OData", edm); 18 19 builder.UseWebApi(config); 20 21 } 22 23 } 24 25 public class Player 26 27 { 28 29 public int Id { get; set; } 30 31 public string Name { get; set; } 32 33 public string Team { get; set; } 34 35 }
允许我们使用所有默认的内建路由约定,例如,
• myapi.com/OData/Players
• myapi.com/OData/Players(key)
• myapi.com/OData/Players(key)/{navigation property | property}
• myapi.com/OData/Players(key)/{function | action}
注意 默认的 Web API OData 路由约定使用了 key 的概念,不是 ID,所以我们 Controller 的 Action 应该接收一个叫做 Key 的参数。
清单 12-6 展示了 ODataController 使用两个 Action 方法。这两个方法都是通过属性路由的方式声明了 OData 路由。
清单 12-6. OData Controller 使用属性路由的例子
1 [ODataRoutePrefix("Players")] 2 3 public class PlayersController : ODataController 4 5 { 6 7 private readonly PlayersContext _players = new PlayersContext(); 8 9 [EnableQuery] 10 11 [ODataRoute] 12 13 public IQueryable<Player> GetAllPlayers() 14 15 { 16 17 return _players.AsQueryable(); 18 19 } 20 21 [EnableQuery] 22 23 [ODataRoute("({key})")] 24 25 public SingleResult<Player> GetSinglePlayers(int key) 26 27 { 28 29 return SingleResult.Create(_players.Where(x => x.Id == key).AsQueryable()); 30 31 } 32 33 }
标签:
版权申明:本站文章部分自网络,如有侵权,请联系:west999com@outlook.com
特别注意:本站所有转载文章言论不代表本站观点,本站所提供的摄影照片,插画,设计作品,如需使用,请与原作者联系,版权归原作者所有
- asp.net源程序编译为dll文件并调用的实现过程 2020-03-29
- Asp.net MVC SignalR来做实时Web聊天实例代码 2020-03-29
- ASP.NET MVC中jQuery与angularjs混合应用传参并绑定数据 2020-03-29
- Asp.Net中WebForm的生命周期 2020-03-29
- ASP.NET使用Ajax返回Json对象的方法 2020-03-23
IDC资讯: 主机资讯 注册资讯 托管资讯 vps资讯 网站建设
网站运营: 建站经验 策划盈利 搜索优化 网站推广 免费资源
网络编程: Asp.Net编程 Asp编程 Php编程 Xml编程 Access Mssql Mysql 其它
服务器技术: Web服务器 Ftp服务器 Mail服务器 Dns服务器 安全防护
软件技巧: 其它软件 Word Excel Powerpoint Ghost Vista QQ空间 QQ FlashGet 迅雷
网页制作: FrontPages Dreamweaver Javascript css photoshop fireworks Flash