PHP框架常见URL模式

在使用PHP框架(CI/TP)搭建项目时, 我们经常使用Nginx/Apache的URL Rewrite模块实现伪静态。例如, 访问URL: http://app.cn/page/13044, 同时我们也可以通过访:http://app.cn/index.php?/page/13044访问,打开的页面是一模一样的。在Nginx设置如下URL Rewrite规则就可以实现:

1
rewrite ^/(.*)$ /index.php?$1 last;

除了伪静态URL外,我们还可以通过其他几种URL模式访问。例如CI框架:

CI框架

CI框架中可选三种URL 模式,在application/config/config.php可以定义。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
/*
|--------------------------------------------------------------------------
| URI PROTOCOL
|--------------------------------------------------------------------------
|
| This item determines which server global should be used to retrieve the
| URI string. The default setting of 'REQUEST_URI' works for most servers.
| If your links do not seem to work, try one of the other delicious flavors:
|
| 'REQUEST_URI' Uses $_SERVER['REQUEST_URI']
| 'QUERY_STRING' Uses $_SERVER['QUERY_STRING']
| 'PATH_INFO' Uses $_SERVER['PATH_INFO']
|
| WARNING: If you set this to 'PATH_INFO', URIs will always be URL-decoded!
*/
$config['uri_protocol'] = 'REQUEST_URI';

REQUEST_URI

REQUEST_URI模式是CI框架默认URL模式,采用分析$_SERVER[‘REQUEST_URI’]来区分控制器/动作/参数。例如:http://localhost/index.php?/user/login?var=value

$_SERVER[‘REQUEST_URI’]值:index.php?/user/login?var=value

实现代码如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
$uri = parse_url('http://dummy'.$_SERVER['REQUEST_URI']);
$query = isset($uri['query']) ? $uri['query'] : '';
$uri = isset($uri['path']) ? $uri['path'] : '';

if (isset($_SERVER['SCRIPT_NAME'][0]))
{
if (strpos($uri, $_SERVER['SCRIPT_NAME']) === 0)
{
$uri = (string) substr($uri, strlen($_SERVER['SCRIPT_NAME']));
}
elseif (strpos($uri, dirname($_SERVER['SCRIPT_NAME'])) === 0)
{
$uri = (string) substr($uri, strlen(dirname($_SERVER['SCRIPT_NAME'])));
}
}

// This section ensures that even on servers that require the URI to be in the query string (Nginx) a correct
// URI is found, and also fixes the QUERY_STRING server var and $_GET array.
if (trim($uri, '/') === '' && strncmp($query, '/', 1) === 0)
{
$query = explode('?', $query, 2);
$uri = $query[0];
$_SERVER['QUERY_STRING'] = isset($query[1]) ? $query[1] : '';
}
else
{
$_SERVER['QUERY_STRING'] = $query;
}

parse_str($_SERVER['QUERY_STRING'], $_GET);

REQUEST_URI模式的基础上添加了重写规则的支持,可以去掉URL地址里面的入口文件index.php

1
2
3
4
5
6
<IfModule mod_rewrite.c> 
RewriteEngine on
RewriteCond %{REQUEST_FILENAME} !-d
RewriteCond %{REQUEST_FILENAME} !-f
ewriteRule ^(.*)$ index.php/$1 [L]
</IfModule>

这样,就可以直接访问:http://localhost/user/login?var=value

QUERY_STRING

QUERT_STRING模式就是传统GET传参数来指定当前访问的模块和操作,例如: `http://localhost/?c=controller&m=method&var=value

c参数表示控制器,m参数表示控制器方法(这些参数都是可以配置的),后面的表示其他GET参数。

1
2
3
$config['enable_query_strings'] = FALSE;
$config['controller_trigger'] = 'c';
$config['function_trigger'] = 'm';

PATH_INFO

PATH-INFO:浏览器用GET方式发送数据时的附加路径

PATHINFO模式下的URL访问地址是: http://localhost/index.php/user/login/var/value/

Apache默认支持PATH_INFO, Nginx则需要改造一下:

1
2
3
4
5
6
7
8
location ~  ^(.+\.php)(.*)$ {
fastcgi_pass 127.0.0.1:9000;
fastcgi_index index.php;
fastcgi_split_path_info ^(.+\.php)(.*)$;
fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name;
fastcgi_param PATH_INFO $fastcgi_path_info;
include fastcgi_params;
}

PATHINFO安全漏洞:http://www.laruence.com/2010/05/20/1495.html (此漏洞已经修复,访问会出现Access denied)

TP

TP(3.2)框架中可以选择四种URL模式。

普通模式

普通模式也就是传统的GET传参方式来指定当前访问的模块和操作,例如: http://localhost/?m=home&c=user&a=login&var=value

m参数表示模块,c参数表示控制器,a参数表示操作(当然这些参数都是可以配置的),后面的表示其他GET参数

如果默认的变量设置和你的应用变量有冲突的话,你需要重新设置系统配置,例如改成下面的:

1
2
3
'VAR_MODULE'            =>  'module',     // 默认模块获取变量
'VAR_CONTROLLER' => 'controller', // 默认控制器获取变量
'VAR_ACTION' => 'action', // 默认操作获取变量

上面的访问地址则变成: http://localhost/?module=home&controller=user&action=login&var=value

PATHINFO模式

PATHINFO模式是TP3.2默认URL模式,PATHINFO模式下面的URL访问地址是: http://localhost/index.php/home/user/login/var/value/

PATHINFO地址的前三个参数分别表示模块/控制器/操作

REWRITE模式

REWRITE模式是在PATHINFO模式的基础上添加了重写规则的支持,可以去掉URL地址里面的入口文件index.php,但是需要额外配置WEB服务器的重写规则。

1
2
3
4
5
6
<IfModule mod_rewrite.c>
RewriteEngine on
RewriteCond %{REQUEST_FILENAME} !-d
RewriteCond %{REQUEST_FILENAME} !-f
RewriteRule ^(.*)$ index.php/$1 [L]
</IfModule>

接下来,就可以用下面的URL地址访问了: http://localhost/home/user/login/var/value

兼容模式

兼容模式是用于不支持PATHINFO的特殊环境,URL地址是: http://localhost/?s=/home/user/login/var/value

兼容模式配合Web服务器重写规则的定义,可以达到和REWRITE模式一样的URL效果。

1
2
3
4
5
6
<IfModule mod_rewrite.c>
RewriteEngine on
RewriteCond %{REQUEST_FILENAME} !-d
RewriteCond %{REQUEST_FILENAME} !-f
RewriteRule ^(.*)$ index.php?s=/$1 [QSA,PT,L]
</IfModule>

就可以和REWRITE模式一样访问下面的URL地址访问了: http://localhost/home/user/login/var/value

参考文档

有用就打赏一下作者吧!