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 查看即可
随机推荐
- Unity Destroy和DestroyImmediate
Destroy(Object obj, float t = 0.0F); 删除一个游戏对象,组件或者资源. 物体obj现在被销毁或在指定了t时间过后销毁.如果obj是组件,它将从GameObject销 ...
- js dairy
//留言js逻辑 $(document).ready( function() { $("#post_btn").click( function() { var comment = ...
- (转)linux 系统下虚拟用户的作用
原文:http://blog.csdn.net/luoshao20120430/article/details/16900653 http://blog.csdn.net/u01 ...
- &&运算符和||运算符的优先级问题 专题
public class SyntaxTest { @Test public void test() { System.out.println(true || true && fals ...
- 用python计算直角三角形斜边长
直接上代码 import math def hypotenuse(a,b): return(math.sqrt(a**2+b**2)) side1 = int(input("第一条直角边:& ...
- cron 任务执行表达式
1.来源 开始我还不知道cron到底来源于哪里,不求甚解的我也没做过多了解,现在突然用到所以写一下. cron计划任务 其实只是linux 一个执行计划的一个工具或者执行程序. 在Linux系统中, ...
- RING3到RING0
当我在说跳转时,说的什么? CPU有很多指令,不是所有的指令都能够随时用,比如 ltr指令就不是随便什么时候能用,在保护模式下,如果你不安规则来执行指令,CPU就会抛出异常,比如你在INTEL手册上就 ...
- AngularJs Type error : Cannot read property 'childNodes' of undefined
参考博客: https://blog.csdn.net/u011127019/article/details/73087868 在AngularJs和JQuery插件共存咋项目中经常会遇到如下异常 T ...
- 《Unity預計算即時GI》笔记:一、基本概念与一些设置
说明 这篇文章是对<Unity預計算即時GI>这个系列文章的笔记. 基本概念 在Unity裡,可以用兩種不同的技術來計算全域光照GI或光源反射,就是烘焙全域光照(Baked GI)和預計算 ...
- intelij idea相关笔记--持续更新
一.快捷键: Ctrl+F 文件内查找 Ctrl+Shift+F 全局查找 Ctrl+Shift+N 查找文件 Ctrl+Alt+← 返回上一步 Ctrl+Alt+→ 返回下一步 二.编译相关: 如果 ...