实施路由选择

路由是指接受用户友好的URL,将URL剖析成它的组成部分,然后做出决定应该调度哪个类和方法的过程。这种实现的好处是,不仅可以使你的URLs搜索引擎优化(SEO)友好,而且可以创建规则,结合正则表达式模式,可以提取参数的值。

如何做...

1.最流行的方法可能是利用支持URL重写的Web服务器。一个例子是配置为使用 mod_rewrite 的Apache网络服务器。然后你定义了重写规则,允许图形文件请求和CSS和JavaScript请求不受影响地通过。否则,请求将通过路由方法被漏过。

2. 另一种潜在的方法是简单地让您的Web服务器虚拟主机定义指向一个特定的路由脚本,然后调用路由类,做出路由决策,并适当地重定向。

3. 第一个要考虑的代码是如何定义路由配置。显而易见的答案是构造一个数组,其中每个键将指向一个正则表达式,URI路径将与之匹配,以及某种形式的操作。下面的代码片段显示了这种配置的一个例子。在这个例子中,我们定义了三个路径:主页、页面和默认值。默认值应该是最后一个,因为它将匹配任何之前没有匹配的内容。这个操作是以匿名函数的形式出现的,如果发生路由匹配,就会被执行。

$config = [
  'home' => [
    'uri' => '!^/$!',
    'exec' => function ($matches) {
      include PAGE_DIR . '/page0.php'; }
  ],
  'page' => [
    'uri' => '!^/(page)/(\d+)$!',
      'exec' => function ($matches) {
        include PAGE_DIR . '/page' . $matches[2] . '.php'; }
  ],
  Router::DEFAULT_MATCH => [
    'uri' => '!.*!',
    'exec' => function ($matches) {
      include PAGE_DIR . '/sorry.php'; }
  ],
];

4. 接下来,我们定义我们的Router类。我们首先定义了在检查和匹配路由的过程中会用到的常量和属性。

namespace Application\Routing;
use InvalidArgumentException;
use Psr\Http\Message\ServerRequestInterface;
class Router
{
  const DEFAULT_MATCH = 'default';
  const ERROR_NO_DEF  = 'ERROR: must supply a default match';
  protected $request;
  protected $requestUri;
  protected $uriParts;
  protected $docRoot;
  protected $config;
  protected $routeMatch;

5. 构造函数接受一个兼容ServerRequestInterface的类、文档根目录的路径和前面提到的配置文件。请注意,如果没有提供默认配置,我们会抛出一个异常。

public function __construct(ServerRequestInterface $request, $docRoot, $config)
{
  $this->config = $config;
  $this->docRoot = $docRoot;
  $this->request = $request;
  $this->requestUri = 
    $request->getServerParams()['REQUEST_URI'];
  $this->uriParts = explode('/', $this->requestUri);
  if (!isset($config[self::DEFAULT_MATCH])) {
      throw new InvalidArgumentException(
        self::ERROR_NO_DEF);
  }
}

6. 接下来,我们有一系列的getter,可以让我们检索原始请求、文档根和最终的路由匹配。

public function getRequest()
{
  return $this->request;
}
public function getDocRoot()
{
  return $this->docRoot;
}
public function getRouteMatch()
{
  return $this->routeMatch;
}

7. isFileOrDir()方法用于确定我们是否正在尝试与CSS、JavaScript或图形请求进行匹配(以及其他可能性)。

public function isFileOrDir()
{
  $fn = $this->docRoot . '/' . $this->requestUri;
  $fn = str_replace('//', '/', $fn);
  if (file_exists($fn)) {
      return $fn;
  } else {
      return '';
  }
}

8. 最后,我们定义match(),它遍历配置数组,并通过preg_match()运行 uri 参数。如果是正值,那么由 preg_match() 填充的配置键和 $matches 数组将存储在 $routeMatch 中,并返回回调。如果没有匹配,则返回默认回调。

public function match()
{
  foreach ($this->config as $key => $route) {
    if (preg_match($route['uri'], 
        $this->requestUri, $matches)) {
        $this->routeMatch['key'] = $key;
        $this->routeMatch['match'] = $matches;
        return $route['exec'];
    }
  }
  return $this->config[self::DEFAULT_MATCH]['exec'];
}
}

如何运行...

首先,改成 /path/to/source/for/this/chapter,并创建一个名为 routing 的目录。接下来,定义一个文件,index.php,它设置了自动加载并使用正确的类。可以定义一个常量PAGE_DIR,指向前面示例中创建的页面目录。

<?php
define('DOC_ROOT', __DIR__);
define('PAGE_DIR', DOC_ROOT . '/../pages');

require_once __DIR__ . '/../../Application/Autoload/Loader.php';
Application\Autoload\Loader::init(__DIR__ . '/../..');
use Application\MiddleWare\ServerRequest;
use Application\Routing\Router;

接下来,添加本示例第3步中讨论的配置数组。请注意,可以在模式的结尾添加(/)?,以说明一个可选的尾部斜杠。另外,对于主页路由,可以提供两个选项://home

$config = [
  'home' => [
    'uri' => '!^(/|/home)$!',
    'exec' => function ($matches) {
      include PAGE_DIR . '/page0.php'; }
  ],
  'page' => [
    'uri' => '!^/(page)/(\d+)(/)?$!',
    'exec' => function ($matches) {
      include PAGE_DIR . '/page' . $matches[2] . '.php'; }
  ],
  Router::DEFAULT_MATCH => [
    'uri' => '!.*!',
    'exec' => function ($matches) {
      include PAGE_DIR . '/sorry.php'; }
  ],
];

然后可以定义一个路由器实例,提供一个初始化的 ServerRequest 实例作为第一个参数。

$router = new Router((new ServerRequest())
  ->initialize(), DOC_ROOT, $config);
$execute = $router->match();
$params  = $router->getRouteMatch()['match'];

然后需要检查请求是文件还是目录,还需要检查路径匹配是否为/

if ($fn = $router->isFileOrDir()
    && $router->getRequest()->getUri()->getPath() != '/') {
    return FALSE;
} else {
    include DOC_ROOT . '/main.php';
}

接下来,定义 main.php,类似这样。

<?php // demo using middleware for routing ?>
<!DOCTYPE html>
<head>
  <title>PHP 7 Cookbook</title>
  <meta http-equiv="content-type" 
  content="text/html;charset=utf-8" />
</head>
<body>
    <?php include PAGE_DIR . '/route_menu.php'; ?>
    <?php $execute($params); ?>
</body>
</html>

最后,还需要修订使用用户友好路由的菜单。

<?php // menu for routing ?>
<a href="/home">Home</a>
<a href="/page/1">Page 1</a>
<a href="/page/2">Page 2</a>
<a href="/page/3">Page 3</a>
<!-- etc. -->

要使用Apache测试配置,定义一个指向 /path/to/source/for/this/chapter/routing 的虚拟主机定义。另外,定义一个.htaccess文件,将任何不是文件、目录或链接的请求导向index.php。另外,您也可以使用内置的PHP webserver。在终端窗口或命令提示符下,键入这个命令。

cd /path/to/source/for/this/chapter/routing
php -S localhost:8080

在浏览器中,当请求http://localhost:8080/home 时的输出是这样的。

另见

关于使用NGINX web服务器重写的信息,请看这篇文章:http://nginx.org/en/docs/http/ngx_http_rewrite_module.html。有很多复杂的 PHP 路由库可以提供比这里介绍的简单路由器更强大的功能。这些包括Altorouter (http://altorouter.com/), TreeRoute (https://github.com/baryshev/TreeRoute), FastRoute (https://github.com/nikic/FastRoute), 和Aura.Router. (https://github.com/auraphp/Aura.Router)。此外,大多数框架(例如,Zend Framework 2或CodeIgniter)都有自己的路由功能。

最后更新于