thinkphp 3.2.3 - Route.class.php 解析(路由匹配)
class Route {
public static function check(){
$depr = C('URL_PATHINFO_DEPR'); // '/'
$regx = preg_replace('/\.'.__EXT__.'$/i','',trim($_SERVER['PATH_INFO'],$depr));
// 分隔符替换 确保路由定义使用统一的分隔符
if('/' != $depr){
$regx = str_replace($depr,'/',$regx);
} // URL映射定义(静态路由)
// http://www.domain.com/?s=pathinfo1
// http://www.domain.com/pathinfo1
$maps = C('URL_MAP_RULES');
// {
// $maps = array(
// "pathinfo1"=>'module1/ctrl1/action1',
// "pathinfo2"=>'module2/ctrl2/action2'
// );
// }
if(isset($maps[$regx])) {
$var = self::parseUrl($maps[$regx]);
// {
// $var[C('VAR_ACTION')] = array_pop($path);
// if(!empty($path)) {
// $var[C('VAR_CONTROLLER')] = array_pop($path);
// }
// if(!empty($path)) {
// $var[C('VAR_MODULE')] = array_pop($path);
// }
// }
$_GET = array_merge($var, $_GET);
return true;
} // 动态路由处理
$routes = C('URL_ROUTE_RULES');
if(!empty($routes)) {
// $routes = array(
// // 写法
// // 'rule1'=>array(0=>'',1=>'',2=>array('ext'=>'.html','method'=>'GET','callback'=>'func1')),
// // array(0=>'rule1',1=>'',2=>array('ext'=>'.html','method'=>'GET','callback'=>'func1')),
// // array(0=>'rule2',1=>'',2=>array('ext'=>'.html','method'=>'POST','callback'=>function(){ return true;/* 匹配成功 return true; */ return false; })), // // 正则路由
// '/[a-z]\/[a-z]\/[a-z]/' => function($module,$ctrl,$action){ return true;/* 匹配成功 return true; */ return false; },
// '/[a-z]\/[a-z]\/[a-z]/'=>array(0=>'',1=>'',2=>array('ext'=>'.html','method'=>'GET','callback'=>function(){ return true;/* 匹配成功 return true; */ return false; })),
// // 内部重写,/ctrl1/xxx/zzz ---> Ctrl1/action1?id=xxx&page=zzz&cate=1 | $_GET = array('c'=>'Ctrl1','a'=>'action1','id'=>'xxx','page'=>'zzz','cate'=>'1','static_param1'=>'static_param1_valu1');
// '/ctrl1\/[a-z]\/[a-z]/'=>array(0=>'Ctrl1/action1?id=:1&page=:2&cate=1',1=>array('static_param1'=>'static_param1_valu1'),2=>array('ext'=>'.html','method'=>'GET','callback'=>function(){ return true;/* 匹配成功 return true; */ return false; })),
// // 内部重写,/ctrl1/xxx/zzz ---> Module1/Ctrl1/action1?id=xxx&page=zzz&cate=1 | $_GET = array('m'=>'Module1','c'=>'Ctrl1','a'=>'action1','id'=>trim('xxx'),'page'=>'zzz','cate'=>'1','static_param1'=>'static_param1_valu1');
// '/ctrl1\/[a-z]\/[a-z]/'=>array(0=>'Module1/Ctrl1/action1?id=:1|trim&page=:2&cate=1',1=>array('static_param1'=>'static_param1_valu1'),2=>array('ext'=>'.html','method'=>'GET','callback'=>function(){ return true;/* 匹配成功 return true; */ return false; })),
// // 内部重写,/ctrl1/xxx/zzz/param1/param1_value ---> Module1/Ctrl1/action1?id=xxx&page=zzz&cate=1 | $_GET = array('m'=>'Module1','c'=>'Ctrl1','a'=>'action1','id'=>trim('xxx'),'page'=>'zzz','cate'=>'1','param1'=>'param1_value','static_param1'=>'static_param1_valu1');
// '/ctrl1\/[a-z]\/[a-z]/'=>array(0=>'Module1/Ctrl1/action1?id=:1|trim&page=:2&cate=1',1=>array('static_param1'=>'static_param1_valu1'),2=>array('ext'=>'.html','method'=>'GET','callback'=>function(){ return true;/* 匹配成功 return true; */ return false; })),
// // 重定向
// '/tobaidu'=>array(0=>'http://www.baidu.com',1=>'301',2=>array('ext'=>'.html','method'=>'GET','callback'=>function(){ return true;/* 匹配成功 return true; */ return false; })),
// '/tobaidu'=>array(0=>'http://www.baidu.com',1=>'302',2=>array('ext'=>'.html','method'=>'GET','callback'=>function(){ return true;/* 匹配成功 return true; */ return false; })),
// '/tourl'=>array(0=>'/desurl',1=>'301',2=>array('ext'=>'.html','method'=>'GET','callback'=>function(){ return true;/* 匹配成功 return true; */ return false; })), // // 规则路由
// '[:param1|trim]'=>function($param1){ return true;/* 匹配成功 return true; */ return false; },
// '[:param1]/[:param2]'=>function($param1,$param2){ return true;/* 匹配成功 return true; */ return false; },
// // 内部重写,/10086/param1/param1_value ---> Module1/Ctrl1/action1?id=10086&cate=1 | $_GET = array('m'=>'Module1','c'=>'Ctrl1','a'=>'action1','id'=>trim('10086'),'cate'=>'1','param1'=>'param1_value','static_param1'=>'static_param1_valu1');
// '[:param1\d|trim]'=>array(0=>'Module1/Ctrl1/action1?id=:1&cate=1',1=>array('static_param1'=>'static_param1_valu1'),2=>array('ext'=>'.html','method'=>'GET','callback'=>function(){ return true;/* 匹配成功 return true; */ return false; })),
// // 内部重写,/xxx/param1/param1_value ---> Module1/Ctrl1/action1?id=xxx&cate=1 | $_GET = array('m'=>'Module1','c'=>'Ctrl1','a'=>'action1','id'=>trim('xxx'),'cate'=>'1','param1'=>'param1_value','static_param1'=>'static_param1_valu1');
// '[:param1^xxx-ctrl2-ctrl3|trim]'=>array(0=>'Module1/Ctrl1/action1?id=:1&cate=1',1=>array('static_param1'=>'static_param1_valu1'),2=>array('ext'=>'.html','method'=>'GET','callback'=>function(){ return true;/* 匹配成功 return true; */ return false; })),
// // 内部重写,/xxx/zzz/param1/param1_value ---> Module1/Ctrl1/action1?id=xxx&page=zzz&cate=1 | $_GET = array('m'=>'Module1','c'=>'Ctrl1','a'=>'action1','id'=>trim('xxx'),'page'=>trim('zzz'),'cate'=>'1','param1'=>'param1_value','static_param1'=>'static_param1_valu1');
// '[:param1^xxx-module2-module3]/[:param1^zzz-ctrl2-ctrl3]/[:param1^action1-action2-action3]'=>array(0=>'Module1/Ctrl1/action1?id=:1&page=:2&cate=1',1=>array('static_param1'=>'static_param1_valu1'),2=>array('ext'=>'.html','method'=>'GET','callback'=>function(){ return true;/* 匹配成功 return true; */ return false; })),
// // 完整匹配,/xxx/zzz ---> Module1/Ctrl1/action1?id=xxx&page=zzz&cate=1 | $_GET = array('m'=>'Module1','c'=>'Ctrl1','a'=>'action1','id'=>trim('xxx'),'page'=>trim('zzz'),'cate'=>'1');
// 'xxx/zzz$'=>array(0=>'Module1/Ctrl1/action1?id=:1&page=:2&cate=1',1=>array('static_param1'=>'static_param1_valu1'),2=>array('ext'=>'.html','method'=>'GET','callback'=>function(){ return true;/* 匹配成功 return true; */ return false; })),
// // 重定向, /xxx/param1/param1_value ---> /desurl
// '[:param1^xxx-module2-module3]'=>array(0=>'/desurl',1=>'301',2=>array('ext'=>'.html','method'=>'GET','callback'=>function(){ return true;/* 匹配成功 return true; */ return false; })),
// );
foreach ($routes as $rule=>$route){
if(is_numeric($rule)){
// 支持 array('rule','adddress',...) 定义路由
$rule = array_shift($route);
}
if(is_array($route) && isset($route[2])){ // 路由参数
$options = $route[2];
if(isset($options['ext']) && __EXT__ != $options['ext']){
// URL后缀检测
continue;
}
if(isset($options['method']) && REQUEST_METHOD != strtoupper($options['method'])){
// 请求类型检测
continue;
}
// 自定义检测
if(!empty($options['callback']) && is_callable($options['callback'])) {
if(false === call_user_func($options['callback'])) {
continue;
}
}
}
if(0===strpos($rule,'/') && preg_match($rule,$regx,$matches)) { // 正则路由
if($route instanceof \Closure) {
// 执行闭包
$result = self::invokeRegx($route, $matches);
// 如果返回布尔值 则继续执行
return is_bool($result) ? $result : exit;
}else{ return self::parseRegex($matches,$route,$regx);
}
}else{ // 规则路由
$len1 = substr_count($regx,'/');
$len2 = substr_count($rule,'/');
if($len1>=$len2 || strpos($rule,'[')) { // “访问路径的长度”大于“规则的长度”
if('$' == substr($rule,-1,1)) {// 完整匹配
if($len1 != $len2) {
continue;
}else{
$rule = substr($rule,0,-1);
}
}
$match = self::checkUrlMatch($regx,$rule);
if(false !== $match) {
if($route instanceof \Closure) {
// 执行闭包
$result = self::invokeRule($route, $match);
// 如果返回布尔值 则继续执行
return is_bool($result) ? $result : exit;
}else{
return self::parseRule($rule,$route,$regx);
}
}
}
}
}
}
return false;
} /**
* “规则路由”检测器
*
* 检测URL和规则路由是否匹配
*/
private static function checkUrlMatch($regx,$rule) {
$m1 = explode('/',$regx); // $_SERVER['PATH_INFO']
$m2 = explode('/',$rule); // module1/ctrl1/act1/[:param1\d]
$var = array();
foreach ($m2 as $key=>$val){
if(0 === strpos($val,'[:')){ // 如: $val == "[:param1\d]"
$val = substr($val,1,-1);
} if(':' == substr($val,0,1)) {// 动态变量 如: $val == ":param1\d"
if($pos = strpos($val,'|')){
// 使用函数过滤
$val = substr($val,1,$pos-1);
}
if(strpos($val,'\\')) { // 如: $val == ":param1\d"
$type = substr($val,-1); // 最后一个字符
if('d'==$type) {
if(isset($m1[$key]) && !is_numeric($m1[$key]))
return false;
}
$name = substr($val, 1, -2); // 变量名称
}elseif($pos = strpos($val,'^')){ // 如: $val == ":param1^ctrl1-ctrl2-ctrl3"
$array = explode('-',substr(strstr($val,'^'),1));
if(in_array($m1[$key],$array)) {
return false;
}
$name = substr($val, 1, $pos - 1);
}else{ // 如: $val == "param1"
$name = substr($val, 1);
}
$var[$name] = isset($m1[$key])?$m1[$key]:'';
}elseif(0 !== strcasecmp($val,$m1[$key])){
return false;
}
}
// 成功匹配后返回URL中的动态变量数组
return $var;
} // 解析规则路由
// '路由规则'=>'[控制器/操作]?额外参数1=值1&额外参数2=值2...'
// '路由规则'=>array('[控制器/操作]','额外参数1=值1&额外参数2=值2...')
// '路由规则'=>'外部地址'
// '路由规则'=>array('外部地址','重定向代码')
// 路由规则中 :开头 表示动态变量
// 外部地址中可以用动态变量 采用 :1 :2 的方式
// 'news/:month/:day/:id'=>array('News/read?cate=1','status=1'),
// 'new/:id'=>array('/new.php?id=:1',301), 重定向
private static function parseRule($rule,$route,$regx) {
// 获取路由地址规则
$url = is_array($route)?$route[0]:$route; // 路由目标的配置
// 获取URL地址中的参数
$paths = explode('/',$regx); // 访问地址
// 解析路由规则
$matches = array();
$rule = explode('/',$rule); // 匹配规则
foreach ($rule as $item){
$fun = '';
if(0 === strpos($item,'[:')){
$item = substr($item,1,-1);
}
if(0===strpos($item,':')) { // 动态变量获取
if($pos = strpos($item,'|')){
// 支持函数过滤
$fun = substr($item,$pos+1);
$item = substr($item,0,$pos);
}
if($pos = strpos($item,'^') ) {
$var = substr($item,1,$pos-1);
}elseif(strpos($item,'\\')){
$var = substr($item,1,-2);
}else{
$var = substr($item,1);
}
$matches[$var] = !empty($fun)? $fun(array_shift($paths)) : array_shift($paths);
}else{ // 过滤URL中的静态变量
array_shift($paths);
}
} if(0=== strpos($url,'/') || 0===strpos($url,'http')) { // 路由重定向跳转
if(strpos($url,':')) { // 传递动态参数
$values = array_values($matches);
$url = preg_replace_callback('/:(\d+)/', function($match) use($values){ return $values[$match[1] - 1]; }, $url);
}
header("Location: $url", true,(is_array($route) && isset($route[1]))?$route[1]:301);
exit;
}else{
// 解析路由地址
$var = self::parseUrl($url);
// 解析路由地址里面的动态参数
$values = array_values($matches);
foreach ($var as $key=>$val){
if(0===strpos($val,':')) {
$var[$key] = $values[substr($val,1)-1];
}
}
$var = array_merge($matches,$var);
// 解析剩余的URL参数
if(!empty($paths)) {
preg_replace_callback('/(\w+)\/([^\/]+)/', function($match) use(&$var){ $var[strtolower($match[1])]=strip_tags($match[2]);}, implode('/',$paths));
}
// 解析路由自动传入参数
if(is_array($route) && isset($route[1])) {
if(is_array($route[1])){
$params = $route[1];
}else{
parse_str($route[1],$params);
}
$var = array_merge($var,$params);
}
$_GET = array_merge($var,$_GET);
}
return true;
}
}
thinkphp 3.2.3 - Route.class.php 解析(路由匹配)的更多相关文章
- thinkphp 3.2.3 - App.class.php 解析
class App { public static function init() { load_ext_file(COMMON_PATH); // { // /home/www/www.domain ...
- Thinkphp源码分析系列(六)–路由机制
在ThinkPHP框架中,是支持URL路由功能,要启用路由功能,需要设置ROUTER_ON 参数为true. 开启路由功能后,系统会自动进行路由检测,如果在路由定义里面找到和当前URL匹配的路由名称, ...
- NET/ASP.NET Routing路由(深入解析路由系统架构原理)(转载)
NET/ASP.NET Routing路由(深入解析路由系统架构原理) 阅读目录: 1.开篇介绍 2.ASP.NET Routing 路由对象模型的位置 3.ASP.NET Routing 路由对象模 ...
- .NET/ASP.NET Routing路由(深入解析路由系统架构原理)
阅读目录: 1.开篇介绍 2.ASP.NET Routing 路由对象模型的位置 3.ASP.NET Routing 路由对象模型的入口 4.ASP.NET Routing 路由对象模型的内部结构 4 ...
- ASP.NET Web API实践系列04,通过Route等特性设置路由
ASP.NET Web API路由,简单来说,就是把客户端请求映射到对应的Action上的过程.在"ASP.NET Web API实践系列03,路由模版, 路由惯例, 路由设置"一 ...
- .NET/ASP.NET Routing路由(深入解析路由系统架构原理)http://wangqingpei557.blog.51cto.com/1009349/1312422
阅读目录: 1.开篇介绍 2.ASP.NET Routing 路由对象模型的位置 3.ASP.NET Routing 路由对象模型的入口 4.ASP.NET Routing 路由对象模型的内部结构 4 ...
- thinkphp URL规则、URL伪静态、URL路由、URL重写、URL生成(十五)
原文:thinkphp URL规则.URL伪静态.URL路由.URL重写.URL生成(十五) 本章节:详细介绍thinkphp URL规则.URL伪静态.URL路由.URL重写.URL生成 一.URL ...
- 一、ASP.NET Routing路由(深入解析路由系统架构原理)
阅读目录: 1.开篇介绍 2.ASP.NET Routing 路由对象模型的位置 3.ASP.NET Routing 路由对象模型的入口 4.ASP.NET Routing 路由对象模型的内部结构 4 ...
- route -A inet6查看路由 getnameinfo failed [UNKNOWN]解决方案
route -A inet6查看路由 getnameinfo failed [UNKNOWN]解决方案, 结果如下: route -A inet6 -n 查看即可
随机推荐
- tcp 三次握手,四次挥手几常见面试题
TCP报文首部 源端口和目的端口,各占2个字节,分别写入源端口和目的端口: 序号,占4个字节,TCP连接中传送的字节流中的每个字节都按顺序编号.例如,一段报文的序号字段值是 301 ,而携带的数据共有 ...
- 一,JVM 自带命令行工具之JPS
jps:虚拟机进程状况工具 可以列出正在运行的虚拟机进程,并显示虚拟机执行主类(main class,class()函数所在的类)的名称,以及这些进程的本地虚拟机的唯一ID. jps命令格式: jps ...
- SQL2008无法启动,报错"17051"解决方法
SQL2008无法启动,这是错误日志: C:/Program Files/Microsoft SQL Server/MSSQL10_50.MSSQLSERVER/MSSQL/Log 2011-06-0 ...
- Shell 学习—AWK介绍
Shell 学习—AWK = = = 安装awk root@kiki-desktop:~/shell# apt-get install gawk gawk-doc = = = awk 是一种程序语言. ...
- JavaScript函数体系
第4章 JavaScript函数 1. 函数基本介绍 ① 为什么需要函数 函数最大的好处就是将零散的代码封装到了一起,当我们要再次使用该功能的时候,不需要再重新书写代码,只需要调用封装好的函数就可以 ...
- JMeter测试TCP服务器遇到的一个奇怪问题
今天工作需要测TCP服务器的压力,因为tsung测试TCP需要写的脚本实在头大,于是换了JMETER来搞压力测试.在实际测试的过程中,遇到了一个很奇怪的问题,就是发了数据包以后,JMeter不停地报5 ...
- tomcat的备份脚本
reference:Crontab的20个例子 先科普一下date的使用方法,在sh脚本中经常会使用得到 date -d<字符串>:显示字符串所指的日期与时间.字符串前后必须加上双引号: ...
- IDEA/AS快捷键收集&习惯
1.Alt+Enter单包引入 2.Ctrl+O (在类中)快速重写父类方法 3.Ctrl+F12显示类结构 4.代码提示 -Ctrl+Alt+空格 代码提示 -Ctrl+Shift+回车 在末尾自动 ...
- JS浏览器获取宽高
screen.availHeight is the height the browser's window can have if it is maximized. (including all th ...
- EEC 欧姆龙PLC输入模块算法
Option Explicit Public MyArray(20000) As Integer Public MyArraySensor(20000) As Integer Sub 生成输入 ...