🔥码云GVP开源项目 12k star Uniapp+ElementUI 功能强大 支持多语言、二开方便! 广告
## thinkphp 5 Url 大小写原理分析 模块目录名不区分大小写,会自动转换为小写 tp5默认URL不区分大小写,默认会自动转换, 控制器 转换规则为: q全部变成小写,首字母转为大写,ac_b转换为AcB 可以关闭自动转换 // 关闭URL自动转换(支持驼峰访问控制器) ~~~ 'url_convert' => false, ~~~ 这样就不会自动转换了,控制器需要写对应正确的控制器名。(规范是大驼峰:首字母大写的驼峰写法) 操作方法的访问本身不会受URL自动转换的影响,但会影响默认的模板渲染输出(这点tp5有待改进,模板应该对应当前方法名才对,并且当前方法名是规范的,小驼峰,所以建议自动转换为小驼峰)。php里面的方法不区分大小写(类也是的) ## thinkphp 3.2.3 URL模式分析 thinkphp 3.2.3的URL_MODEL配置项可以配置四种URL模式(默认为PATHINFO模式) - 0 - 普通模式 - 1 - pathinfo - 2 - REWRITE模式 - 3 - 兼容模式 ~~~ // URL 模式定义 const URL_COMMON = 0; //普通模式 const URL_PATHINFO = 1; //PATHINFO模式 const URL_REWRITE = 2; //REWRITE模式 const URL_COMPAT = 3; // 兼容模式 ~~~ **1,2,3可以设置伪静态后缀,生成url时默认会有后缀(URL_HTML_SUFFIX伪静态后缀设置)** >[info] 后缀只是使url看起来在更像静态地址而已,并没有什么实际作用(实际上也有一些高级用法,比如接口开发中常用后缀约定返回数据类型,或者特殊的请求约定,参见 [路由章节](http://ihavenolimitations.xyz/xiak/php-node/227302)) > >[danger] 注意:后缀`__EXT__`只有在`$_SERVER['PATH_INFO']`存在是才有,也就是说,普通模式没有后缀,兼容模式也可以有后缀,因为兼容模式其实是通过将兼容参数赋的值给了`$_SERVER['PATH_INFO']`来模拟实现pathinfo。 - 0 - 没什么好说的就是最原始的url - 1 - 默认 index.php/home/index/index 没有隐藏index.php - 2 - REWRITE模式 在1模式的基础上隐藏了index.php,但是需要Apache和nginx支持REWRITE(url重写,伪静态模块), >[info] 通过url重写可以隐藏index.php,这样U函数生成url就隐藏index.php了(其实1,2模式U函数这里是一样的,主要是因为\__APP\__不同)。URL重写实际上是利用了pathinfo,如果是老版本的nginx不支持pathinfo的话,则需要将重写规则换一种方式,其实是利用3兼容模式(?s=home/index/index)来实现隐藏入口。 - 3 - 如果既不支持pathinfo,也不支持Url重写,又不愿意将就普通模式,那还有一个办法,就是兼容模式,兼容模式的形式是 ?s=home/index/index ,处理请求时程序内部会做兼容处理,将$_GET['s']赋值给$_SERVER['PATH_INFO'],只有这点不同,U函数生成URL时与1,2模式是一样的。 >[info] 其实上面这些只是url常规基本的处理,只是美化url而已,更高级的自定义url请阅读 [路由章节](http://ihavenolimitations.xyz/xiak/php-node/227302) ### 下面请参考关键代码部分: **Dispatcher.class.php 关键代码** ~~~ $varPath = C('VAR_PATHINFO'); …… if(isset($_GET[$varPath])) { // 判断URL里面是否有兼容模式参数 $_SERVER['PATH_INFO'] = $_GET[$varPath]; unset($_GET[$varPath]); }elseif(IS_CLI){ // CLI模式下 index.php module/controller/action/params/... $_SERVER['PATH_INFO'] = isset($_SERVER['argv'][1]) ? $_SERVER['argv'][1] : ''; } …… if(!defined('__APP__')){ $urlMode = C('URL_MODEL'); if($urlMode == URL_COMPAT ){// 兼容模式判断 define('PHP_FILE',_PHP_FILE_.'?'.$varPath.'='); }elseif($urlMode == URL_REWRITE ) { // REWRITE模式 $url = dirname(_PHP_FILE_); if($url == '/' || $url == '\\') $url = ''; define('PHP_FILE',$url); }else { // 普通模式和PATHINFO模式 define('PHP_FILE',_PHP_FILE_); } // 当前应用地址 define('__APP__',strip_tags(PHP_FILE)); } ~~~ **function文件U函数关键代码** ~~~ if(C('URL_MODEL') == 0) { // 普通模式URL转换 $url = __APP__.'?'.C('VAR_MODULE')."={$module}&".http_build_query(array_reverse($var)); if($urlCase){ $url = strtolower($url); } if(!empty($vars)) { $vars = http_build_query($vars); $url .= '&'.$vars; } }else{ // PATHINFO模式或者兼容URL模式 if(isset($route)) { $url = __APP__.'/'.rtrim($url,$depr); }else{ $module = (defined('BIND_MODULE') && BIND_MODULE==$module )? '' : $module; $url = __APP__.'/'.($module?$module.MODULE_PATHINFO_DEPR:'').implode($depr,array_reverse($var)); } if($urlCase){ $url = strtolower($url); } if(!empty($vars)) { // 添加参数 foreach ($vars as $var => $val){ if('' !== trim($val)) $url .= $depr . $var . $depr . urlencode($val); } } if($suffix) { $suffix = $suffix===true?C('URL_HTML_SUFFIX'):$suffix; if($pos = strpos($suffix, '|')){ $suffix = substr($suffix, 0, $pos); } if($suffix && '/' != substr($url,-1)){ $url .= '.'.ltrim($suffix,'.'); } } } ~~~ ## 针对的thinkphp项目的伪静态配置 **Apache** httpd.conf配置文件中加载了mod_rewrite.so模块 AllowOverride None 将None改为 All 把下面的内容保存为.htaccess文件放到应用入口文件的同级目录下 ~~~ <IfModule mod_rewrite.c> Options +FollowSymlinks -Multiviews RewriteEngine on RewriteCond %{REQUEST_FILENAME} !-d RewriteCond %{REQUEST_FILENAME} !-f RewriteRule ^(.*)$ index.php?/$1 [QSA,PT,L] </IfModule> ~~~ **Nginx** 在Nginx低版本中,是不支持PATHINFO的,但是可以通过在Nginx.conf中配置转发规则实现: ~~~ location / { // …..省略部分代码 if (!-e $request_filename) { rewrite ^(.*)$ /index.php?s=/$1 last; break; } } ~~~ 参考: - [什么是伪静态](http://ihavenolimitations.xyz/xiak/php-node/227302) - [伪静态](http://ihavenolimitations.xyz/manual/thinkphp5/118046) - [URL重写](http://ihavenolimitations.xyz/manual/thinkphp5/177576) ## 扩展 pathinfo 参数 和 get参数可以共存,但只有一种url形式可以,下面来看: 实例代码 ~~~php <?php echo '<pre>'; echo '<h2>REQUEST_URI</h2>'; var_dump($_SERVER['REQUEST_URI']); echo '<br />'; echo '<h2>PATH_INFO</h2>'; var_dump($_SERVER['PATH_INFO']); echo '<br />'; echo '<h2>$_GET</h2>'; print_r($_GET); echo '<br />'; echo '<h2>$_SERVER</h2>'; print_r($_SERVER); echo '</pre>'; ~~~ 输出: ~~~html REQUEST_URI string(29) "/b.php/a/b/c.html?a=1&b=2&c=3" PATH_INFO string(11) "/a/b/c.html" $_GET Array ( [a] => 1 [b] => 2 [c] => 3 ) $_SERVER Array ( [HTTP_HOST] => 127.0.0.1 [HTTP_CONNECTION] => keep-alive [HTTP_CACHE_CONTROL] => max-age=0 [HTTP_ACCEPT] => text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,*/*;q=0.8 [HTTP_UPGRADE_INSECURE_REQUESTS] => 1 [HTTP_USER_AGENT] => Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/45.0.2454.101 Safari/537.36 [HTTP_ACCEPT_ENCODING] => gzip, deflate, sdch [HTTP_ACCEPT_LANGUAGE] => zh-CN,zh;q=0.8 [HTTP_COOKIE] => HTsp_2132_saltkey=eNJelI18; HTsp_2132_lastvisit=1477874651; HTsp_2132_visitedfid=2; HTsp_2132_nofavfid=1; HTsp_2132_editormode_e=1; HTsp_2132_ulastactivity=e175EkmLd6OZui%2FDf0PuF0TAjLZV%2B0aO8X6pFBlliEi49cxPRdif; HTsp_2132_smile=1D1; HTsp_2132_widthauto=-1 [PATH] => C:\Windows\system32;C:\Windows;C:\Windows\System32\Wbem;C:\Windows\System32\WindowsPowerShell\v1.0\;D:\Program Files\Git\cmd;C:\Program Files (x86)\AMD\ATI.ACE\Core-Static;D:\Program Files\nodejs\; [SystemRoot] => C:\Windows [COMSPEC] => C:\Windows\system32\cmd.exe [PATHEXT] => .COM;.EXE;.BAT;.CMD;.VBS;.VBE;.JS;.JSE;.WSF;.WSH;.MSC [WINDIR] => C:\Windows [SERVER_SIGNATURE] => Apache/2.4.23 (Win64) PHP/5.6.25 Server at 127.0.0.1 Port 80 [SERVER_SOFTWARE] => Apache/2.4.23 (Win64) PHP/5.6.25 [SERVER_NAME] => 127.0.0.1 [SERVER_ADDR] => 127.0.0.1 [SERVER_PORT] => 80 [REMOTE_ADDR] => 127.0.0.1 [DOCUMENT_ROOT] => D:/wamp64/www [REQUEST_SCHEME] => http [CONTEXT_PREFIX] => [CONTEXT_DOCUMENT_ROOT] => D:/wamp64/www [SERVER_ADMIN] => wampserver@wampserver.invalid [SCRIPT_FILENAME] => D:/wamp64/www/b.php [REMOTE_PORT] => 52676 [GATEWAY_INTERFACE] => CGI/1.1 [SERVER_PROTOCOL] => HTTP/1.1 [REQUEST_METHOD] => GET [QUERY_STRING] => a=1&b=2&c=3 [REQUEST_URI] => /b.php/a/b/c.html?a=1&b=2&c=3 [SCRIPT_NAME] => /b.php [PATH_INFO] => /a/b/c.html [PATH_TRANSLATED] => D:\wamp64\www\a.php\b\c.html [PHP_SELF] => /b.php/a/b/c.html [REQUEST_TIME_FLOAT] => 1478592492.192 [REQUEST_TIME] => 1478592492 ) ~~~