热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

[WebAPI]WebAPI2深入系列(5)特性路由

小分享:我有几张阿里云优惠券,用券购买或者升级阿里云相应产品最多可以优惠五折!领券地址:https:promotion.aliyun.comntmsactambassadorsha

小分享:我有几张阿里云优惠券,用券购买或者升级阿里云相应产品最多可以优惠五折!领券地址:https://promotion.aliyun.com/ntms/act/ambassador/sharetouser.html?userCode=ohmepe03


目录

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 previousInitializer = configuration.Initializer;
    configuration.Initializer = (Action) (cOnfig=>
    {
        previousInitializer(config);
        aggregateRoute.EnsureInitialized((Func>) (() =>
        {
            subRoutes = new SubRouteCollection();
            AttributeRoutingMapper.AddRouteEntries(subRoutes, configuration, constraintResolver, directRouteProvider);
            return subRoutes;
        }));
    });
}

RouteCollectionRoute是特性路由的HttpRoute对象,既是一个IHttpRoute对象,又是一个IHttpRoute集合.并且其中核心方法为GetRouteData(IHttpRoute其他接口都返回为null),

internal class RouteCollectionRoute : IHttpRoute, IReadOnlyCollection, IEnumerable, IEnumerable
{
    public IHttpRouteData GetRouteData(string virtualPathRoot, HttpRequestMessage request)
    {
      List httpRouteDataList = new List();
      //调用内部的SubRoutes对象
      foreach (IHttpRoute subRoute in (IEnumerable) 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, IEnumerable, IEnumerable
{
    private readonly List _routes = new List();
    private readonly List _entries = new List();
    public IReadOnlyCollection Entries{get;}
}

而SubRoutes的创建是在MapHttpAttributeRoutes方法定义,实际调用是在HttpServer的Send方法初始化的.

protected override async Task SendAsync(HttpRequestMessage request, CancellationToken cancellationToken)
{
    this.EnsureInitialized();
    //...
}

首先验证一下我们的特性路由注册位置(定义1个扩展方法)

public static class RouteCollectionExt
{
    public static IEnumerable 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;
        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 FindMatchingActions(HttpControllerContext controllerContext, bool ignoreVerbs = false)
{
    //此处做特性路由判断
    IEnumerable subRoutes = controllerContext.RouteData.GetSubRoutes();
    return subRoutes == null ? 普通路由 : 特性路由;
}

备注:
- 如果我们为特性路由指定了Name,则会自动创建一个IHttpRoute绑定到RouteCollection上.(这步是在HttpConfiguration初始化中最后做判断完成的)

- 文章中的代码并非完整WebAPI代码,一般是经过自己精简后的.

- 本篇内容使用MarkDown语法编辑

首发地址:http://neverc.cnblogs.com/p/5975086.html


参考页面:http://qingqi 标签" href="http://www.yuanjiaocheng.net/Jsp/jstl-core-remove-tag.html" target="_blank">ngquege.cnblogs.com/p/5933752.html

[Web API] Web API 2 深入系列(5) 特性路由


推荐阅读
  • 本文介绍了lua语言中闭包的特性及其在模式匹配、日期处理、编译和模块化等方面的应用。lua中的闭包是严格遵循词法定界的第一类值,函数可以作为变量自由传递,也可以作为参数传递给其他函数。这些特性使得lua语言具有极大的灵活性,为程序开发带来了便利。 ... [详细]
  • HDU 2372 El Dorado(DP)的最长上升子序列长度求解方法
    本文介绍了解决HDU 2372 El Dorado问题的一种动态规划方法,通过循环k的方式求解最长上升子序列的长度。具体实现过程包括初始化dp数组、读取数列、计算最长上升子序列长度等步骤。 ... [详细]
  • 本文介绍了C#中数据集DataSet对象的使用及相关方法详解,包括DataSet对象的概述、与数据关系对象的互联、Rows集合和Columns集合的组成,以及DataSet对象常用的方法之一——Merge方法的使用。通过本文的阅读,读者可以了解到DataSet对象在C#中的重要性和使用方法。 ... [详细]
  • 本文介绍了OC学习笔记中的@property和@synthesize,包括属性的定义和合成的使用方法。通过示例代码详细讲解了@property和@synthesize的作用和用法。 ... [详细]
  • 本文介绍了在SpringBoot中集成thymeleaf前端模版的配置步骤,包括在application.properties配置文件中添加thymeleaf的配置信息,引入thymeleaf的jar包,以及创建PageController并添加index方法。 ... [详细]
  • 知识图谱——机器大脑中的知识库
    本文介绍了知识图谱在机器大脑中的应用,以及搜索引擎在知识图谱方面的发展。以谷歌知识图谱为例,说明了知识图谱的智能化特点。通过搜索引擎用户可以获取更加智能化的答案,如搜索关键词"Marie Curie",会得到居里夫人的详细信息以及与之相关的历史人物。知识图谱的出现引起了搜索引擎行业的变革,不仅美国的微软必应,中国的百度、搜狗等搜索引擎公司也纷纷推出了自己的知识图谱。 ... [详细]
  • 本文详细介绍了Linux中进程控制块PCBtask_struct结构体的结构和作用,包括进程状态、进程号、待处理信号、进程地址空间、调度标志、锁深度、基本时间片、调度策略以及内存管理信息等方面的内容。阅读本文可以更加深入地了解Linux进程管理的原理和机制。 ... [详细]
  • 后台获取视图对应的字符串
    1.帮助类后台获取视图对应的字符串publicclassViewHelper{将View输出为字符串(注:不会执行对应的ac ... [详细]
  • 《数据结构》学习笔记3——串匹配算法性能评估
    本文主要讨论串匹配算法的性能评估,包括模式匹配、字符种类数量、算法复杂度等内容。通过借助C++中的头文件和库,可以实现对串的匹配操作。其中蛮力算法的复杂度为O(m*n),通过随机取出长度为m的子串作为模式P,在文本T中进行匹配,统计平均复杂度。对于成功和失败的匹配分别进行测试,分析其平均复杂度。详情请参考相关学习资源。 ... [详细]
  • Windows下配置PHP5.6的方法及注意事项
    本文介绍了在Windows系统下配置PHP5.6的步骤及注意事项,包括下载PHP5.6、解压并配置IIS、添加模块映射、测试等。同时提供了一些常见问题的解决方法,如下载缺失的msvcr110.dll文件等。通过本文的指导,读者可以轻松地在Windows系统下配置PHP5.6,并解决一些常见的配置问题。 ... [详细]
  • Mac OS 升级到11.2.2 Eclipse打不开了,报错Failed to create the Java Virtual Machine
    本文介绍了在Mac OS升级到11.2.2版本后,使用Eclipse打开时出现报错Failed to create the Java Virtual Machine的问题,并提供了解决方法。 ... [详细]
  • 在说Hibernate映射前,我们先来了解下对象关系映射ORM。ORM的实现思想就是将关系数据库中表的数据映射成对象,以对象的形式展现。这样开发人员就可以把对数据库的操作转化为对 ... [详细]
  • 本文讲述了作者通过点火测试男友的性格和承受能力,以考验婚姻问题。作者故意不安慰男友并再次点火,观察他的反应。这个行为是善意的玩人,旨在了解男友的性格和避免婚姻问题。 ... [详细]
  • 1,关于死锁的理解死锁,我们可以简单的理解为是两个线程同时使用同一资源,两个线程又得不到相应的资源而造成永无相互等待的情况。 2,模拟死锁背景介绍:我们创建一个朋友 ... [详细]
  • 动态规划算法的基本步骤及最长递增子序列问题详解
    本文详细介绍了动态规划算法的基本步骤,包括划分阶段、选择状态、决策和状态转移方程,并以最长递增子序列问题为例进行了详细解析。动态规划算法的有效性依赖于问题本身所具有的最优子结构性质和子问题重叠性质。通过将子问题的解保存在一个表中,在以后尽可能多地利用这些子问题的解,从而提高算法的效率。 ... [详细]
author-avatar
白纸鹤u8_fw7
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有