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 解析(路由匹配)的更多相关文章

  1. thinkphp 3.2.3 - App.class.php 解析

    class App { public static function init() { load_ext_file(COMMON_PATH); // { // /home/www/www.domain ...

  2. Thinkphp源码分析系列(六)–路由机制

    在ThinkPHP框架中,是支持URL路由功能,要启用路由功能,需要设置ROUTER_ON 参数为true. 开启路由功能后,系统会自动进行路由检测,如果在路由定义里面找到和当前URL匹配的路由名称, ...

  3. NET/ASP.NET Routing路由(深入解析路由系统架构原理)(转载)

    NET/ASP.NET Routing路由(深入解析路由系统架构原理) 阅读目录: 1.开篇介绍 2.ASP.NET Routing 路由对象模型的位置 3.ASP.NET Routing 路由对象模 ...

  4. .NET/ASP.NET Routing路由(深入解析路由系统架构原理)

    阅读目录: 1.开篇介绍 2.ASP.NET Routing 路由对象模型的位置 3.ASP.NET Routing 路由对象模型的入口 4.ASP.NET Routing 路由对象模型的内部结构 4 ...

  5. ASP.NET Web API实践系列04,通过Route等特性设置路由

    ASP.NET Web API路由,简单来说,就是把客户端请求映射到对应的Action上的过程.在"ASP.NET Web API实践系列03,路由模版, 路由惯例, 路由设置"一 ...

  6. .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 ...

  7. thinkphp URL规则、URL伪静态、URL路由、URL重写、URL生成(十五)

    原文:thinkphp URL规则.URL伪静态.URL路由.URL重写.URL生成(十五) 本章节:详细介绍thinkphp URL规则.URL伪静态.URL路由.URL重写.URL生成 一.URL ...

  8. 一、ASP.NET Routing路由(深入解析路由系统架构原理)

    阅读目录: 1.开篇介绍 2.ASP.NET Routing 路由对象模型的位置 3.ASP.NET Routing 路由对象模型的入口 4.ASP.NET Routing 路由对象模型的内部结构 4 ...

  9. route -A inet6查看路由 getnameinfo failed [UNKNOWN]解决方案

    route -A inet6查看路由 getnameinfo failed [UNKNOWN]解决方案, 结果如下: route -A inet6 -n 查看即可

随机推荐

  1. 记录一个在线压缩和还原压缩js代码的工具

    packer – javascript 压缩工具 http://dean.edwards.name/packer/ Javascript Beautifier ---可以恢复某些压缩工具压缩的js代码 ...

  2. Linux利用iptables实现真-全局代理

    对于经常要浏览油管等被墙网站的人而言,利用代理来实现fq是非常有必要的.现在fq的方法中,最为主流的应该要数ssr了,因此本教程都是基于ssr的socks5代理而言的. 在windows中,ssr客户 ...

  3. 你还在为UiPath课程考试发愁吗?

    刚开始学UiPath的时候,课程的考试难倒了很多人,有语言的原因也有对课程理解的原因,记忆中好像有一课考了5次估计,由于题库也就那么多,只要你努力考,总会过的. 学会了RPA的自动化工具,能否自动化答 ...

  4. Likely root cause: java.lang.IllegalStateException: jar hell!

    jar hell 解决方案: 问题: Likely root cause: java.lang.IllegalStateException: jar hell! 解决: 当一一个类或者一一个资源文件存 ...

  5. VS2015配置使用Sqlite以及EF6框架记录

    项目中需要使用到Sqlite本地数据库保存数据,以防止离线情况下设备的正常使用. 一.下载vs2015下的sqlite插件,并安装 插件下载页面:http://system.data.sqlite.o ...

  6. Java并发编程的艺术,解读并发编程的优缺点

    并发编程的优缺点 使用并发的原因 多核的CPU的背景下,催生了并发编程的趋势,通过并发编程的形式可以将多核CPU的计算能力发挥到极致,性能得到提升. 在特殊的业务场景下先天的就适合于并发编程. 比如在 ...

  7. git从安装到应用的学习

    本文内容是对廖雪峰老师Git教程做的笔记,外加一些自己的学习心得,还抱着学以致用的心态来实践一番 Git学习笔记 分布式版本控制与集中式版本控制的最大区别 集中式的版本控制,本地没有历史记录,完整的仓 ...

  8. cmd对其他盘符进行操作

    一般我们打开cmd命令时,会出现如下界面: 现在,我想要对g盘进行操作,则输入 --> g:,然后回车,如图: 可以查看一下g盘下的所有子目录,输入 --> dir,回车,结果如下: 我想 ...

  9. JavaScript中函数声明和函数表达式的区别

    声明一个函数: var x=1; foo(); function foo() { console.log(x);//1 } myfun();//报错 定义一个函数表达式: var myfun=myfo ...

  10. Browser Window

    Window 对象 Window对象表示浏览器中打开的窗口. 如果文档包含框架(iframe或iframe标签),浏览器会被html文档创建一个window对象,并为每个框架创建一个额外的window ...