喜讯!TCMS 官网正式上线!一站式提供企业级定制研发、App 小程序开发、AI 与区块链等全栈软件服务,助力多行业数智转型,欢迎致电:13888011868 QQ 932256355 洽谈合作!
本文聚焦 PHP 内置服务器 URL 重写,提供轻量本地开发方案。核心是通过自定义路由脚本,遵循 “静态资源优先 + 规则分组排序” 原则,实现与 Nginx/Apache 等效功能,无需依赖重型服务器。 文章明确启动命令参数顺序,详解项目结构、服务器启动及调试配置,提供兼容 PHP 5.6 + 的基础路由脚本,解决静态资源 404、规则失效等痛点。进阶部分以 ThinkPHP 为例,用 “精准→通用” 规则分组适配复杂需求,梳理常见问题解决方案。该方案可直接复用,适配主流 PHP 框架,减少环境差异问题,提升开发效率。

/bazijp.html 映射为 /?ac=bazijp ),通常依赖Nginx/Apache的 rewrite 指令。但本地开发时,PHP内置服务器( php -S )更轻量高效,仅需通过自定义路由脚本即可实现等效重写功能。本文将从基础原理、环境配置、静态资源兼容、复杂规则适配(如ThinkPHP,laravel项目)等维度,结合实际项目的重写需求,提供一套可直接复用的解决方案,兼容PHP 5.6+主流版本。
PHP内置服务器本身不支持类似Nginx的 rewrite 语法,需通过路由脚本(如 router.php )拦截所有请求并自定义转发逻辑,核心流程如下:
客户端发起请求(如 sm.tekin.cn/bazijp.html );
内置服务器将请求优先转发至路由脚本;
路由脚本根据预设规则重写URL(如映射为 sm.tekin.cn/index.php?ac=bazijp );
重写后的请求转发至项目统一入口(如 public/index.php );
若请求为静态资源(CSS/JS/图片),则直接返回文件,不进入重写流程。
关键前提 :内置服务器对命令行参数顺序有严格要求,必须遵循 php -S [地址:端口] -t [文档根目录] [路由脚本] ,参数错位会导致路由失效或静态资源无法加载。
以常见的“Web根目录与业务代码分离”结构为例(如ThinkPHP、Laravel等框架通用),目录结构如下:
project-root/ # 项目根目录
├─ public/ # Web根目录(线上访问根路径)
│ ├─ statics/ # 静态资源目录(CSS/JS/图片)
│ └─ index.php # 项目统一入口文件
└─ router.php # URL重写路由脚本
在 项目根目录 执行命令,指定 public 为Web根目录, router.php 为路由脚本,确保与线上环境目录映射一致:
# 基础启动命令(本地访问地址:http://127.0.0.1:8000)
php -S 127.0.0.1:8000 -t public router.php
# 启用Xdebug调试(兼容PHP 5.6+)
php -dxdebug.remote_enable=1 -dxdebug.remote_autostart=1 -S 127.0.0.1:8000 -t public router.php若需在IDE中调试业务逻辑,需配置 launch.json 确保调试与重写协同生效,关键在于参数顺序与线上一致:
{
"version": "0.2.0",
"configurations": [
{
"name": "PHP内置服务器+URL重写+调试",
"type": "php",
"runtimeExecutable": "/opt/local/bin/php56", // 本地PHP路径(需与项目版本匹配)
"request": "launch",
"runtimeArgs": [
"-dxdebug.remote_enable=1",
"-dxdebug.remote_autostart=1",
"-dxdebug.remote_port=9000", // 与php.ini中Xdebug端口一致
"-S", "127.0.0.1:8000",
"-t", "public", // 匹配线上Web根目录
"router.php" // 路由脚本必须放在最后
],
"cwd": "${workspaceRoot}",
"serverReadyAction": {
"pattern": "Development Server \\(http://localhost:([0-9]+)\\) started",
"uriFormat": "http://localhost:%s",
"action": "openExternally" // 启动后自动打开本地调试地址
}
}
]
}本地开发中,两类问题最为常见:
静态资源(如 /statics/ffsm/global.css )被路由脚本拦截,返回404;
简单重写规则(如 /hehun.html → /?ac=hehun )不生效,无法访问目标模块。
核心逻辑: 优先处理静态资源,再执行URL重写 ,确保静态资源加载与动态路由分离:
<?php
// 项目根目录与Web根目录定义
$projectRoot = __DIR__;
$webRoot = rtrim($projectRoot . '/public', '/') . '/';
// 1. 解析并标准化请求URI
$requestUri = $_SERVER['REQUEST_URI'];
$uriParts = parse_url($requestUri);
$uriPath = isset($uriParts['path']) ? $uriParts['path'] : '/';
$uriPath = preg_replace('#/\.\./#', '/', $uriPath); // 过滤目录遍历攻击
$uriPath = rtrim($uriPath, '/'); // 统一去除末尾斜杠(如`/suanming/scbz/`→`/suanming/scbz`)
$uriPath = $uriPath === '' ? '/' : $uriPath;
// 2. 优先处理静态资源(存在则直接返回)
$requestedFile = $webRoot . ltrim($uriPath, '/');
if (file_exists($requestedFile) && is_file($requestedFile) && !is_dir($requestedFile)) {
// 设置正确MIME类型,避免浏览器解析异常
$mimeType = getMimeType($requestedFile);
if ($mimeType) header("Content-Type: {$mimeType}");
readfile($requestedFile);
exit;
}
// 3. 定义基础URL重写规则(可根据项目需求扩展)
$rewriteRules = [
'#^/index\.html$#' => '/index.php', // 首页规则
'#^/bazijp\.html$#' => '/?ac=bazijp', // 八字精批模块规则
'#^/hehun\.html$#' => '/?ac=hehun', // 合婚模块规则
'#^/aboutus\.html$#' => '/?ac=aboutus', // 关于我们页面规则
'#^/xyd-([0-9]+)\.html$#' => '/?ac=xyd&id=$1', // 详情页动态规则
'#^/([^/]+)\.html$#' => '/index.php?ac=$1', // 最后执行:通用.html规则(最宽泛,避免覆盖前面的具体规则)
];
// 4. 应用重写规则
$newUri = $uriPath;
foreach ($rewriteRules as $pattern => $target) {
if (preg_match($pattern, $uriPath)) {
$newUri = preg_replace($pattern, $target, $uriPath);
break; // 匹配到即终止,避免规则冲突
}
}
// 5. 处理查询参数(保留原参数,新规则参数覆盖同名原参数)
$originalQuery = isset($uriParts['query']) ? $uriParts['query'] : '';
$newUriParts = parse_url($newUri);
$newPath = isset($newUriParts['path']) ? $newUriParts['path'] : '/';
$newQuery = isset($newUriParts['query']) ? $newUriParts['query'] : '';
$finalQuery = '';
if (!empty($originalQuery) && !empty($newQuery)) {
parse_str($originalQuery, $originalParams);
parse_str($newQuery, $newParams);
$mergedParams = array_merge($originalParams, $newParams);
$finalQuery = http_build_query($mergedParams);
} elseif (!empty($originalQuery)) {
$finalQuery = $originalQuery;
} else {
$finalQuery = $newQuery;
}
// 6. 更新服务器变量,转发到统一入口
$finalUri = $newPath . ($finalQuery ? "?{$finalQuery}" : '');
$_SERVER['REQUEST_URI'] = $finalUri;
$_SERVER['SCRIPT_NAME'] = '/index.php';
$_SERVER['QUERY_STRING'] = $finalQuery;
parse_str($finalQuery, $_GET); // 同步更新GET参数,适配框架参数获取
// 7. 执行入口文件
$indexFile = $webRoot . 'index.php';
if (file_exists($indexFile)) {
include_once $indexFile;
} else {
http_response_code(404);
echo "404 Not Found:public/index.php 入口文件不存在";
}
exit;
/**
* 兼容PHP 5.6的MIME类型获取
* @param string $file 文件路径
* @return string|null MIME类型
*/
function getMimeType($file) {
$extension = strtolower(pathinfo($file, PATHINFO_EXTENSION));
$mimeMap = [
'css' => 'text/css', 'js' => 'application/javascript',
'html' => 'text/html', 'jpg' => 'image/jpeg',
'png' => 'image/png', 'gif' => 'image/gif', 'ico' => 'image/x-icon'
];
return isset($mimeMap[$extension]) ? $mimeMap[$extension] : null;
}访问 http://127.0.0.1:8000/bazijp.html ,通过 var_dump($_GET) 应看到 array('ac' => 'bazijp') ;
访问 http://127.0.0.1:8000/xyd-123.html ,应看到 array('ac' => 'xyd', 'id' => '123') ;
访问 http://127.0.0.1:8000/statics/ffsm/global.css ,应直接返回CSS文件内容。
实际项目中,常需处理大量复杂重写规则(如多模块路由、动态参数拼接)。例如某命理类项目的Nginx规则片段:
rewrite ^/aboutus.html /index.php?ac=aboutus last;
rewrite ^/xyd-([0-9]+).html /index.php?ac=xyd&id=$1 last;
rewrite ^/(.*).html /index.php?ac=$1 last;
rewrite ^/show-([0-9]+).html /index.php?ct=news&ac=show&id=$1;这类规则迁移时,核心挑战是 避免通用规则覆盖具体规则 (如 /aboutus.html 不能被 /.+\.html 的通用规则错误映射)。
核心思路:将规则按“精准度”分组, 精准规则优先匹配 ,通用规则兜底,确保每个模块的路由逻辑与线上一致。
<?php
$projectRoot = __DIR__;
$webRoot = rtrim($projectRoot . '/public', '/') . '/';
// 1. 标准化请求URI
$requestUri = $_SERVER['REQUEST_URI'];
$uriParts = parse_url($requestUri);
$uriPath = isset($uriParts['path']) ? $uriParts['path'] : '/';
$uriPath = preg_replace('#/\.\./#', '/', $uriPath);
$uriPath = rtrim($uriPath, '/');
$uriPath = $uriPath === '' ? '/' : $uriPath;
// 2. 优先处理静态资源
$requestedFile = $webRoot . ltrim($uriPath, '/');
if (file_exists($requestedFile) && is_file($requestedFile) && !is_dir($requestedFile)) {
$mimeType = getMimeType($requestedFile);
if ($mimeType) header("Content-Type: {$mimeType}");
readfile($requestedFile);
exit;
}
// 3. 规则分组:精准规则 → 通用规则(避免宽覆盖窄)
// 注意下面的规则需要根据你自己的项目进行修改,这里仅做示例 更多可以参考 https://sm.tekin.cn 站点的URL重写
// --------------------------
// 第一组:精准规则(无动态参数,固定URL)
// --------------------------
$exactRules = [
// 基础页面
'#^/index\.html$#' => '/index.php',
'#^/aboutus\.html$#' => '/index.php?ac=aboutus',
'#^/contactus\.html$#' => '/index.php?ac=contactus',
];
// --------------------------
// 第二组:通用规则(带动态参数、后缀匹配)
// --------------------------
$generalRules = [
// 带ID的精准后缀规则
'#^/xyd-([0-9]+)\.html$#' => '/index.php?ac=xyd&id=$1',
// 姓名模块动态路径
'#^/xmfx/([^/]+)$#' => '/index.php?ct=xingming&ac=xmfx&name=$1',
'#^/xqlist-([0-9]+)-([0-9]+)-([0-9]+)-([0-9]+)\.html$#' => '/index.php?ct=xq&xid=$1&sex=$2&hs=$3&page=$4',
// 最后执行:通用.html规则(兜底)
'#^/([^/]+)\.html$#' => '/index.php?ac=$1',
];
// 4. 执行重写:先精准后通用
$newUri = $uriPath;
$ruleMatched = false;
// 第一步:匹配精准规则(核心业务优先)
foreach ($exactRules as $pattern => $target) {
if (preg_match($pattern, $uriPath)) {
$newUri = preg_replace($pattern, $target, $uriPath);
$ruleMatched = true;
break;
}
}
// 第二步:精准规则未匹配时,匹配通用规则
if (!$ruleMatched) {
foreach ($generalRules as $pattern => $target) {
if (preg_match($pattern, $uriPath)) {
$newUri = preg_replace($pattern, $target, $uriPath);
break;
}
}
}
// 5. 合并查询参数(同基础版逻辑)
$originalQuery = isset($uriParts['query']) ? $uriParts['query'] : '';
$newUriParts = parse_url($newUri);
$newPath = isset($newUriParts['path']) ? $newUriParts['path'] : '/';
$newQuery = isset($newUriParts['query']) ? $newUriParts['query'] : '';
$finalQuery = '';
if (!empty($originalQuery) && !empty($newQuery)) {
parse_str($originalQuery, $originalParams);
parse_str($newQuery, $newParams);
$mergedParams = array_merge($originalParams, $newParams);
$finalQuery = http_build_query($mergedParams);
} elseif (!empty($originalQuery)) {
$finalQuery = $originalQuery;
} else {
$finalQuery = $newQuery;
}
// 6. 转发到入口文件
$finalUri = $newPath . ($finalQuery ? "?{$finalQuery}" : '');
$_SERVER['REQUEST_URI'] = $finalUri;
$_SERVER['SCRIPT_NAME'] = '/index.php';
$_SERVER['QUERY_STRING'] = $finalQuery;
parse_str($finalQuery, $_GET);
$indexFile = $webRoot . 'index.php';
if (file_exists($indexFile)) {
include_once $indexFile;
} else {
http_response_code(404);
echo "404 Not Found:public/index.php 入口文件不存在";
}
exit;
// MIME类型函数(同基础版)
function getMimeType($file) {
$extension = strtolower(pathinfo($file, PATHINFO_EXTENSION));
$mimeMap = [
'css' => 'text/css', 'js' => 'application/javascript',
'html' => 'text/html', 'jpg' => 'image/jpeg',
'png' => 'image/png', 'gif' => 'image/gif', 'ico' => 'image/x-icon'
];
return isset($mimeMap[$extension]) ? $mimeMap[$extension] : null;
}
?>分组逻辑 : $exactRules 存放固定URL(如 /aboutus.html ), $generalRules 存放动态URL(如 /([^/]+)\.html ),确保精准规则不被覆盖;
通用规则内部顺序 :即使在 $generalRules 中,也需从“具体”到“宽泛”排序(如先匹配 /xyd-([0-9]+)\.html ,再匹配 /([^/]+)\.html );
参数兼容性 :保留原请求中的查询参数(如 /user/abc?foo=bar → /index.php?ct=user&ac=abc&foo=bar ),符合线上重写习惯。
原因 :路由脚本未优先处理静态资源,或 $webRoot 路径拼接错误(如多写斜杠);
解决方案 :确保静态资源判断逻辑在重写规则之前, $requestedFile 路径格式为 public/statics/ffsm/global.css 。
原因 :内置服务器参数顺序错误(如 router.php 放在 -t public 之前),或正则表达式错误(如 . 未转义);
解决方案 :严格遵循 php -S 地址 -t 根目录 路由脚本 ,正则使用 #^/aboutus\.html$# 格式。
问题 :使用 ?? 空合并运算符导致语法错误;
解决方案 :用 isset() +三元运算符替代(如 $uriPath = isset($uriParts['path']) ? $uriParts['path'] : '/' )。
PHP内置服务器的URL重写核心在于路由脚本的设计,通过“静态资源优先+规则分组排序”的思路,可实现与Nginx/Apache等效的重写功能。关键要点如下:
参数顺序 :启动时严格遵循 -S 地址 -t 根目录 路由脚本 ;
静态优先 :避免静态资源进入重写流程;
规则分组 :按“精准→通用”排序,解决规则覆盖问题;
版本兼容 :针对PHP 5.6等旧版本调整语法,确保项目可用性。