侯体宗的博客
  • 首页
  • Hyperf版
  • beego仿版
  • 人生(杂谈)
  • 技术
  • 关于我
  • 更多分类
    • 文件下载
    • 文字修仙
    • 中国象棋ai
    • 群聊
    • 九宫格抽奖
    • 拼图
    • 消消乐
    • 相册

PHP路由库FastRoute的使用教程

php  /  管理员 发布于 7年前   380

GitHub: https://github.com/nikic/FastRoute

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

安装

通过 composer 安装

composer require nikic/fast-route

要求 PHP 5.4 及更高的版本

使用

这是一个基本的使用示例

<?phprequire '/path/to/vendor/autoload.php';$dispatcher = FastRoute\simpleDispatcher(function(FastRoute\RouteCollector $r) {    $r->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'];// 去除查询字符串( ? 后面的内容) 和 解码 URIif (false !== $pos = strpos($uri, '?')) {    $uri = substr($uri, 0, $pos);}$uri = rawurldecode($uri);$routeInfo = $dispatcher->dispatch($httpMethod, $uri);switch ($routeInfo[0]) {    case FastRoute\Dispatcher::NOT_FOUND:        // ... 404 Not Found 没找到对应的方法        break;    case FastRoute\Dispatcher::METHOD_NOT_ALLOWED:        $allowedMethods = $routeInfo[1];        // ... 405 Method Not Allowed  方法不允许        break;    case FastRoute\Dispatcher::FOUND: // 找到对应的方法        $handler = $routeInfo[1]; // 获得处理函数        $vars = $routeInfo[2]; // 获取请求参数        // ... call $handler with $vars // 调用处理函数        break;}

定义路由

通过调用 FastRoute\simpleDispatcher() 函数来定义路由,该函数接受一个以 FastRoute\RouteCollector 实例为参数的闭包作为参数。通过在 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,可以缓存生成的路由数据并从缓存的信息构建调度。

<?php$dispatcher = FastRoute\cachedDispatcher(function(FastRoute\RouteCollector $r) {    $r->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_FOUND、Dispatcher::METHOD_NOT_ALLOWED、Dispatcher::FOUND 其中之一。对于 Dispatcher::METHOD_NOT_ALLOWED 状态,第二个数组元素包含允许提供的 URI 的 HTTP 方法列表。

[FastRoute\Dispatcher::METHOD_NOT_ALLOWED, ['GET', 'POST']]

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

/* Routing against GET /user/nikic/42 */[FastRoute\Dispatcher::FOUND, 'handler0', ['name' => 'nikic', 'id' => '42']]

重写路由解析器和调度器

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

<?phpnamespace FastRoute;interface RouteParser {    public function parse($route);}interface DataGenerator {    public function addRoute($httpMethod, $routeData, $handler);    public function getData();}interface Dispatcher {    const NOT_FOUND = 0, FOUND = 1, METHOD_NOT_ALLOWED = 2;    public function dispatch($httpMethod, $uri);}

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

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

然后可以将该数组传递给数据生成器的 addRoute() 方法,在添加了所有路由之后,调用生成器的 getData(),它将返回调度器所需的所有路由数据。

调度程序通过构造函数接受路由数据,并提供 dispatch()方法。

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

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

<?php$dispatcher = FastRoute\simpleDispatcher(function(FastRoute\RouteCollector $r) {    /* ... */}, [    'routeParser' => 'FastRoute\\RouteParser\\Std',    'dataGenerator' => 'FastRoute\\DataGenerator\\GroupCountBased',    'dispatcher' => 'FastRoute\\Dispatcher\\GroupCountBased',]);

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

关于HEAD请求的说明

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

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

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

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

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

推荐教程:《PHP教程》

以上就是PHP路由库FastRoute的使用教程的详细内容,更多请关注其它相关文章!


  • 上一条:
    PHP中的预定义变量的个人见解
    下一条:
    PHP简单实现路由Route功能
  • 昵称:

    邮箱:

    0条评论 (评论内容有缓存机制,请悉知!)
    最新最热
    • 分类目录
    • 人生(杂谈)
    • 技术
    • linux
    • Java
    • php
    • 框架(架构)
    • 前端
    • ThinkPHP
    • 数据库
    • 微信(小程序)
    • Laravel
    • Redis
    • Docker
    • Go
    • swoole
    • Windows
    • Python
    • 苹果(mac/ios)
    • 相关文章
    • Laravel从Accel获得5700万美元A轮融资(0个评论)
    • PHP 8.4 Alpha 1现已发布!(0个评论)
    • 用Time Warden监控PHP中的代码处理时间(0个评论)
    • 在PHP中使用array_pop + yield实现读取超大型目录功能示例(0个评论)
    • Property Hooks RFC在PHP 8.4中越来越接近现实(0个评论)
    • 近期文章
    • 智能合约Solidity学习CryptoZombie第三课:组建僵尸军队(高级Solidity理论)(0个评论)
    • 智能合约Solidity学习CryptoZombie第二课:让你的僵尸猎食(0个评论)
    • 智能合约Solidity学习CryptoZombie第一课:生成一只你的僵尸(0个评论)
    • 在go中实现一个常用的先进先出的缓存淘汰算法示例代码(0个评论)
    • 在go+gin中使用"github.com/skip2/go-qrcode"实现url转二维码功能(0个评论)
    • 在go语言中使用api.geonames.org接口实现根据国际邮政编码获取地址信息功能(1个评论)
    • 在go语言中使用github.com/signintech/gopdf实现生成pdf分页文件功能(0个评论)
    • gmail发邮件报错:534 5.7.9 Application-specific password required...解决方案(0个评论)
    • 欧盟关于强迫劳动的规定的官方举报渠道及官方举报网站(0个评论)
    • 在go语言中使用github.com/signintech/gopdf实现生成pdf文件功能(0个评论)
    • 近期评论
    • 122 在

      学历:一种延缓就业设计,生活需求下的权衡之选中评论 工作几年后,报名考研了,到现在还没认真学习备考,迷茫中。作为一名北漂互联网打工人..
    • 123 在

      Clash for Windows作者删库跑路了,github已404中评论 按理说只要你在国内,所有的流量进出都在监控范围内,不管你怎么隐藏也没用,想搞你分..
    • 原梓番博客 在

      在Laravel框架中使用模型Model分表最简单的方法中评论 好久好久都没看友情链接申请了,今天刚看,已经添加。..
    • 博主 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 @1111老铁这个不行了,可以看看近期评论的其他文章..
    • 1111 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 网站不能打开,博主百忙中能否发个APP下载链接,佛跳墙或极光..
    • 2016-10
    • 2016-11
    • 2017-06
    • 2017-07
    • 2017-08
    • 2017-09
    • 2017-11
    • 2017-12
    • 2018-01
    • 2018-02
    • 2018-03
    • 2020-03
    • 2020-04
    • 2020-05
    • 2020-06
    • 2020-07
    • 2020-09
    • 2021-02
    • 2021-03
    • 2021-04
    • 2021-05
    • 2021-06
    • 2021-07
    • 2021-08
    • 2021-09
    • 2021-10
    • 2021-11
    • 2021-12
    • 2022-01
    • 2022-02
    • 2022-05
    • 2022-06
    • 2022-07
    • 2022-08
    • 2022-09
    • 2022-10
    • 2022-11
    • 2022-12
    • 2023-01
    • 2023-02
    • 2023-03
    • 2023-04
    • 2023-05
    • 2023-06
    • 2023-07
    • 2023-08
    • 2023-09
    • 2023-10
    • 2023-11
    • 2023-12
    • 2024-01
    • 2024-02
    • 2024-03
    • 2024-04
    • 2024-05
    • 2024-06
    • 2024-07
    • 2024-09
    Top

    Copyright·© 2019 侯体宗版权所有· 粤ICP备20027696号 PHP交流群

    侯体宗的博客