PHP ルーティング ライブラリ FastRoute の使用に関するチュートリアル

Guanhui
リリース: 2023-04-08 15:24:02
転載
4130 人が閲覧しました

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

这个库提供了基于正则表达式的快速路由实现。这篇文章解释了 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 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,如何解释它取决于你。

请求方法的书写快捷方式

对于GETPOSTPUTPATCHDELETEHEAD请求方法,可使用快捷方式。

$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 方法列表。

[FastRoute\Dispatcher::METHOD_NOT_ALLOWED, ['GET', 'POST']]
ログイン後にコピー

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

/* Routing against GET /user/nikic/42 */ [FastRoute\Dispatcher::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时,可以通过传入额外的参数,进行覆盖

 'FastRoute\\RouteParser\\Std', 'dataGenerator' => 'FastRoute\\DataGenerator\\GroupCountBased', 'dispatcher' => 'FastRoute\\Dispatcher\\GroupCountBased', ]);
ログイン後にコピー

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

关于HEAD请求的说明

HTTP 规范要求服务器 同时支持GETHEAD方法

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

ユーザーがリソースごとにHEADルートを手動で登録することを強制されないようにするため、リクエストは一致するGETルートで応答されます。 PHP Web SAPI は、HEAD応答からエンティティ本体を透過的に削除するため、この動作は大部分のユーザーに影響を与えません。

ただし、Web SAPI 環境の外で FastRoute を使用する場合は、HEADリクエストに応じて生成されたエンティティ本文を決して送信してはなりません。SAPI 以外のユーザーの場合は、これを送信してください。ただし、FastRoute には HTTP の破壊を制限する力はありません。

最後に、アプリケーションは常に、特定のリソースに対して独自のHEADメソッド ルートを指定して、この動作を完全にバイパスできることに注意してください。

推奨チュートリアル: 「PHP チュートリアル

以上がPHP ルーティング ライブラリ FastRoute の使用に関するチュートリアルの詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

関連ラベル:
php
ソース:csdn.net
このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。
最新のダウンロード
詳細>
ウェブエフェクト
公式サイト
サイト素材
フロントエンドテンプレート
私たちについて 免責事項 Sitemap
PHP中国語ウェブサイト:福祉オンライン PHP トレーニング,PHP 学習者の迅速な成長を支援します!