资讯专栏INFORMATION COLUMN

FastRoute - 快速请求路由

mist14 / 902人阅读

链接
https://github.com/nikic/Fast...

这个库提供了基于正则表达式的快速路由实现。这篇文章解释了 FastRoute 是如何工作的和它为什么很快。

安装

通过 composer 安装

composer require nikic/fast-route

要求 PHP 5.4 及更高的版本

使用

这是一个基本的使用示例

addRoute("GET", "/users", "get_all_users_handler");
    // {id} 必须是一个数字 (d+)
    $r->addRoute("GET", "/user/{id:d+}", "get_user_handler");
    //  /{title} 后缀是可选的
    $r->addRoute("GET", "/articles/{id:d+}[/{title}]", "get_article_handler");
});

// 获取请求的方法和 URI
$httpMethod = $_SERVER["REQUEST_METHOD"];
$uri = $_SERVER["REQUEST_URI"];

// 去除查询字符串( ? 后面的内容) 和 解码 URI
if (false !== $pos = strpos($uri, "?")) {
    $uri = substr($uri, 0, $pos);
}
$uri = rawurldecode($uri);

$routeInfo = $dispatcher->dispatch($httpMethod, $uri);
switch ($routeInfo[0]) {
    case FastRouteDispatcher::NOT_FOUND:
        // ... 404 Not Found 没找到对应的方法
        break;
    case FastRouteDispatcher::METHOD_NOT_ALLOWED:
        $allowedMethods = $routeInfo[1];
        // ... 405 Method Not Allowed  方法不允许
        break;
    case FastRouteDispatcher::FOUND: // 找到对应的方法
        $handler = $routeInfo[1]; // 获得处理函数
        $vars = $routeInfo[2]; // 获取请求参数
        // ... call $handler with $vars // 调用处理函数
        break;
}
定义路由

通过调用 FastRoutesimpleDispatcher() 函数来定义路由,该函数接受一个以 FastRouteRouteCollector 实例为参数的闭包作为参数。通过在 collector 实例里面调用 addRoute() 增加路由。

$r->addRoute($method, $routePattern, $handler);

$method 是大写的 HTTP 方法,能够被某个路由匹配,可以使用数组指定多个有效的 $method 。

// 这里两行调用
$r->addRoute("GET", "/test", "handler");
$r->addRoute("POST", "/test", "handler");
// 等同于这一行调用
$r->addRoute(["GET", "POST"], "/test", "handler");

默认情况下 $routePattern 使用一种语法,比如 {foo} 是指定名称为 foo 的占位符,可以匹配正则表达式 [^/]+. 。要调整占位符匹配的模式,可以通过编写 {bar:[0-9] +} 来指定自定义模式。一些例子

// 匹配 /user/42,不匹配 /user/xyx
$r->addRoute("GET", "/user/{id:d+}", "handler");

// 匹配 /user/foobar,不匹配 /user/foo/bar
$r->addRoute("GET", "/user/{name}", "handler");

// 匹配 /user/foobar,也匹配 /user/foo/bar
$r->addRoute("GET", "/user/{name:.+}", "handler");

路由占位符的自定义模式不能使用捕获组,例如 {lang:(en|de)} 不是有效的占位符,因为 () 是一个捕获组,可以使用 {lang:en|de} 或者 {lang:(?:en|de)} 代替。

另外,在路由 [...] 中定义的部分是可选匹配的,所以 /foo[bar] 将匹配 /foo/foobar 。路由可选部分只支持在定义的末尾,而不能在定义的中间。

// 这个路由有,[/{name}] 可选择匹配部分
$r->addRoute("GET", "/user/{id:d+}[/{name}]", "handler");
// 等同于这两个路由
$r->addRoute("GET", "/user/{id:d+}", "handler");
$r->addRoute("GET", "/user/{id:d+}/{name}", "handler");

// 多层嵌套可选路由,也是支持的
$r->addRoute("GET", "/user[/{id:d+}[/{name}]]", "handler");

// 这个路由定义无效,因为可选部分只能在定义的末尾
$r->addRoute("GET", "/user[/{id:d+}]/{name}", "handler");

$handler 参数不一定必须是回调函数,它也可以是控制器类名或任何其他类型的数据。FastRoute 只告诉你哪个 handler 对应 URI,如何解释它取决于你。

请求方法的书写快捷方式

对于 GET、POST、PUT、PATCH、DELETE 和 HEAD 请求方法,可使用快捷方式。

$r->get("/get-route", "get_handler");
$r->post("/post-route", "post_handler");

// 等同于
$r->addRoute("GET", "/get-route", "get_handler");
$r->addRoute("POST", "/post-route", "post_handler");
路由组

你可以在一个组内定义路由,同一组内的路由有相同的前缀。

$r->addGroup("/admin", function (RouteCollector $r) {
    $r->addRoute("GET", "/do-something", "handler");
    $r->addRoute("GET", "/do-another-thing", "handler");
    $r->addRoute("GET", "/do-something-else", "handler");
});

// 等同于
$r->addRoute("GET", "/admin/do-something", "handler");
$r->addRoute("GET", "/admin/do-another-thing", "handler");
$r->addRoute("GET", "/admin/do-something-else", "handler");

可以定义多层嵌套组结构。

缓存

使用 simpleDispatcher 定义路由的回调函数可以无缝缓存。通过使用 cachedDispatcher 而不是 simpleDispatcher,可以缓存生成的路由数据并从缓存的信息构建调度。

addRoute("GET", "/user/{name}/{id:[0-9]+}", "handler0");
    $r->addRoute("GET", "/user/{id:[0-9]+}", "handler1");
    $r->addRoute("GET", "/user/{name}", "handler2");
}, [
    "cacheFile" => __DIR__ . "/route.cache", /* required 缓存文件路径,必须设置 */
    "cacheDisabled" => IS_DEBUG_ENABLED,     /* optional, enabled by default 是否缓存,可选参数,默认情况下开启 */
]);

该函数的第二个参数是一个选项数组,可用于指定缓存文件路径等等。

调度 URI

通过调用 dispatch() 调度 URI。这个方法接受 HTTP 方法 和一个 URI 作为参数。获得这两个信息是你自己的工作,这个库并不绑定到 PHP web SAPIs 。
dispatch() 返回一个数组,第一个元素是一个状态码,状态码是 Dispatcher::NOT_FOUNDDispatcher::METHOD_NOT_ALLOWEDDispatcher::FOUND 其中之一。对于 Dispatcher::METHOD_NOT_ALLOWED 状态,第二个数组元素包含允许提供的 URI 的 HTTP 方法列表。

[FastRouteDispatcher::METHOD_NOT_ALLOWED, ["GET", "POST"]]

对于 Dispatcher::FOUND 状态,第二个数组元素是 $handler ,第三个数组元素是是一个包含所有占位符的数组

/* Routing against GET /user/nikic/42 */

[FastRouteDispatcher::FOUND, "handler0", ["name" => "nikic", "id" => "42"]]
重写路由解析器和调度器

这个库使用三个组件,一个路由解析器,一个数据生成器,一个调度器。这个三个组件实现以下接口


路由解析器获取路由模式字符串并将其转换为路由信息数组,其中每个路线信息又是它的部分数组。

/* The route /user/{id:d+}[/{name}] converts to the following array: */
[
    [
        "/user/",
        ["id", "d+"],
    ],
    [
        "/user/",
        ["id", "d+"],
        "/",
        ["name", "[^/]+"],
    ],
]

然后可以将该数组传递给数据生成器的 addRoute() 方法,在添加了所有路由之后,调用生成器的 getData(),它将返回调度器所需的所有路由数据。
调度程序通过构造函数接受路由数据,并提供 dispatch()方法。

路由解析器可以被多带带覆盖,然而数据生成器和调度器应该总是一起修改,因为前者的输出与后者的输入紧密耦合。

当使用 simpleDispatcher / cachedDispatcher 时,可以通过传入额外的参数,进行覆盖

 "FastRouteRouteParserStd",
    "dataGenerator" => "FastRouteDataGeneratorGroupCountBased",
    "dispatcher" => "FastRouteDispatcherGroupCountBased",
]);

上面给出了默认的设置,通过把 GroupCountBased 替换成 GroupPosBased 可以使用完全不同的调度策略

关于HEAD请求的说明

HTTP 规范要求服务器 同时支持 GET 和 HEAD 方法

GET和HEAD方法必须得到所有通用服务器的支持

为避免强制用户为每个资源手动注册 HEAD 路由,将使用一个匹配的 GET 路由响应请求。PHP web SAPI 透明地从 HEAD 响应中移除实体主体,所以这种行为对绝大多数用户没有影响。

但是,在 Web SAPI 环境外部使用 FastRoute ,绝不能发送响应 HEAD 请求而生成的实体主体,如果你是非 SAPI 用户,这是你的责任;在这种情况下,FastRoute 无权限制你破坏 HTTP 。

最后,请注意,应用程序可以始终为给定资源指定其自己的 HEAD 方法路由以完全绕过此行为。

总结

文档还是很好理解,下次就要看源码了。

原创文章,欢迎转载。转载请注明出处,谢谢。
原文链接地址:http://dryyun.com/2018/04/20/...
作者: dryyun
发表日期: 2018-04-21 10:14:21

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/28662.html

相关文章

  • 翻译 使用正则的快速路由

    摘要:的路由是使用了这个库作者写了一篇帖子介绍了它写这个库的原因原文链接使用正则的快速路由库前段时间我在路由库遇到了一些问题。它号称比现用的路由库快几个数量级的,因为为了达到这个这个目的,这个库是通过的扩展实现的。 首先先区分一下概念:路由是指一个过程,就是利用定义好的一些规则,让不同的URI能够调用不同的处理器(一个匿名函数或者一个类中的方法)这样一个过程。 平常很多框架所说的定义一个路由...

    Lionad-Morotar 评论0 收藏0
  • PHP路由性能测试

    摘要:路由控制器性能测试前言前段时间抽空写了个微型路由控制器可以在我的另一篇文章里面看到详细的介绍。上门有测试以及这几个路由控制器的性能,总体来说和在最坏的情况下表现最好。同样的测试条件。所以本次测试仅仅针对,以及。 PHP路由控制器性能测试 前言 前段时间抽空写了个微型路由控制器可以在我的另一篇文章里面看到详细的介绍。虽然在设计的时候及尽量避开了正则匹配和数组循环这种耗时的操作。尽量节省时...

    levy9527 评论0 收藏0
  • Just for fun——基于Swoole做个小框架

    摘要:使开发人员可以编写高性能的异步并发,服务。使用作为网络通信框架,可以使企业研发团队的效率大大提升,更加专注于开发创新产品。总之,这个库让可以常驻内存,并提供了,等功能。 swoole 使 PHP 开发人员可以编写高性能的异步并发 TCP、UDP、Unix Socket、HTTP,WebSocket 服务。Swoole 可以广泛应用于互联网、移动通信、企业软件、云计算、网络游戏、物联网(...

    CoreDump 评论0 收藏0
  • Just for fun——基于Swoole做个小框架

    摘要:使开发人员可以编写高性能的异步并发,服务。使用作为网络通信框架,可以使企业研发团队的效率大大提升,更加专注于开发创新产品。总之,这个库让可以常驻内存,并提供了,等功能。 swoole 使 PHP 开发人员可以编写高性能的异步并发 TCP、UDP、Unix Socket、HTTP,WebSocket 服务。Swoole 可以广泛应用于互联网、移动通信、企业软件、云计算、网络游戏、物联网(...

    fevin 评论0 收藏0
  • 如何使用 Zend Expressive 建立 NASA 图片库?

    摘要:在本文中,我们将借助天文图库,使用建立图片库。在使用虚拟机时,此处应为,命令将在目录下运行。我们建议在选择服务名时,尽量使用完整的类名。这样,相当于告诉它必须使用指定的类来创建服务。在返回中的最后一个响应之前,应用会缓存该响应以备下次使用。 在本文中,我们将借助 NASA 天文图库 API,使用 Zend Expressive 建立图片库。最后的结果将显示在 AstroSpla...

    hidogs 评论0 收藏0

发表评论

0条评论

最新活动
阅读需要支付1元查看
<