当前位置:首页 > PHP教程 > Zend > 列表

Zend Framework框架路由机制代码分析

发布:smiling 来源: PHP粉丝网  添加日期:2021-07-20 22:16:09 浏览: 评论:0 

这篇文章主要介绍了Zend Framework框架路由机制,结合代码分析了Zend Framework路由机制所涉及的控制器,路由器原理与相关代码实现技巧,需要的朋友可以参考下。

本文分析了Zend Framework框架路由机制代码,分享给大家供大家参考,具体如下:

在框架中,有关路由的调用关系为:

1、apache的mod_rewrite模块把请求路由到框架的启动脚本,一般是index.php;

2、前端控制器Zend_Controller_Front通过dispatch函数进行请求分发;

3、路由器Zend_Controller_Router_Rewrite通过route函数处理路由,对路由器中已有的路由规则,按照加入顺序的逆序(类似于栈,后进先出)对每个route调用match函数,以检查请求是否和当前路由规则匹配,如果匹配的话把路由器的当前路由这个变量($_currentRoute)设置为匹配的路由,并把route解析出来的参数传给Zend_Controller_Request_Http对象,到这里完成路由设置。

如果没有发现路由,框架会使用Index控制器的index这个action。

对Zend_Controller_Router_Route中的函数代码分析:

1、构造函数

  1. public function __construct($route$defaults = array(), $reqs = array()) 
  2.   $route = trim($route$this->_urlDelimiter); //去掉规则首尾的url分隔符(默认是/) 
  3.   $this->_defaults = (array$defaults//默认值数组,以变量名为键 
  4.   $this->_requirements = (array$reqs//变量需要满足的正则表达式,以变量名为键 
  5.   if ($route != '') { 
  6.    foreach (explode($this->_urlDelimiter, $routeas $pos => $part) { 
  7.     //把规则切分为一个数组 
  8.     if (substr($part, 0, 1) == $this->_urlVariable) {//如果是一个变量的定义 
  9.      $name = substr($part, 1); //获取变量名 
  10.      //如果该变量定义了对应的正则表达式,则获取该表达式,否则置为null 
  11.      $regex = (isset($reqs[$name]) ? $reqs[$name] : $this->_defaultRegex); 
  12.      //_parts数组包含了规则的各个部分,如果是变量的话,数组中有name元素 
  13.      $this->_parts[$pos] = array('name' => $name'regex' => $regex); 
  14.      //_vars包含了该规则中的所有变量的名字 
  15.      $this->_vars[] = $name
  16.     } else { //普通字符串 
  17.      $this->_parts[$pos] = array('regex' => $part); 
  18.      if ($part != '*') { 
  19.       $this->_staticCount++; //该规则的普通字符串的个数 
  20.      } 
  21.     } 
  22.    } 
  23.   } 

2、匹配算法

  1. public function match($path
  2.   $pathStaticCount = 0; 
  3.   $defaults = $this->_defaults; //默认值数组,数组元素的键值是变量名 
  4.    //默认值数组的一个拷贝,不过变量的值全部换成布尔值,其实这个值并没有实际用处,下面程序仅仅 
  5.   //是通过判断键值是否存在而确定是否包含一个变量,可能这么做是为了节省空间,不过要是这样的话 
  6.   //不如直接使用 $this->_defaults了? 
  7.   if (count($defaults)) { 
  8.    $unique = array_combine(array_keys($defaults), array_fill(0, count($defaults), true)); 
  9.   } else { 
  10.    $unique = array(); 
  11.   } 
  12.   $path = trim($path$this->_urlDelimiter); //传入的path是已经去掉baseUrl的,这里确保去掉首尾的分隔符 
  13.   if ($path != '') { 
  14.    $path = explode($this->_urlDelimiter, $path); 
  15.    foreach ($path as $pos => $pathPart) { 
  16.     if (!isset($this->_parts[$pos])) { 
  17.      //把path根据url分隔符分割为数组后,把每一部分和规则的对应部分比较,如果path中存在, 
  18.      //而规则中不存在对应部分,那么该规则肯定不匹配,这里要注意$pos,是通过它把规则 
  19.      //和path的对应部分对应起来。 
  20.      return false; 
  21.     } 
  22.     if ($this->_parts[$pos]['regex'] == '*') { 
  23.       //如果规则的当前部分是通配符*,则把path的剩余部分解释为url传递过来的变量,他们按照 
  24.      //“变量名/变量值”这样的形式成对出现 
  25.      $parts = array_slice($path$pos); //获取path的剩余部分 
  26.      $this->_getWildcardData($parts$unique); 
  27.      break
  28.     } 
  29.     $part = $this->_parts[$pos]; 
  30.     $name = isset($part['name']) ? $part['name'] : null; 
  31.     $pathPart = urldecode($pathPart);//对传过来的值进行解码 
  32.     if ($name === null) {//普通字符串,和规则的对应部分比较是否相等即可 
  33.      if ($part['regex'] != $pathPart) { 
  34.       return false; 
  35.      } 
  36.     } elseif ($part['regex'] === null) { 
  37.       //如果是变量,但是没有需要满足的正则表达式,那么只有值不为空就可以了 
  38.      if (strlen($pathPart) == 0) { 
  39.       return false; 
  40.      } 
  41.     } else {//如果对该变量需要满足一个正则表达式,那么这里进行验证 
  42.      $regex = $this->_regexDelimiter . '^' . $part['regex'] . '$' . $this->_regexDelimiter . 'iu'
  43.      if (!preg_match($regex$pathPart)) { 
  44.       return false; 
  45.      } 
  46.     } 
  47.     if ($name !== null) { 
  48.      // 如果是一个变量,则设置变量的值 
  49.      $this->_values[$name] = $pathPart
  50.      $unique[$name] = true; //其实没有必要设置,这个版本根本就没有用它 
  51.     } else { 
  52.      //把普通字符串的匹配计数加1,因为规则中的普通字符串是必须在path中存在的,否则就是 
  53.      //匹配失败 
  54.      $pathStaticCount++; 
  55.     } 
  56.    } 
  57.   } 
  58.    //$this->_values中保存的是分析获取的变量,如果规则中存在‘*',则$this->_params是获取的 
  59.   //变量,否则是空数组,$this->_defaults是规则提供的默认变量值,这里用‘+'把三个数组相加 
  60.   //这样的好处是如果后面的数组与前面的数组有相同的非整数的键值,后面的不会覆盖前面的,这 
  61.   //与array_merge函数有区别,后者是会覆盖的。也就是说,如果$this->_values 中已经有键controller 
  62.   //,那么$this->_defaults中的controller元素就被忽略,这样就$this->_defaults中的默认值只有在path 
  63.   //中不存在的时候才会出现在返回值中。 
  64.   $return = $this->_values + $this->_params + $this->_defaults; 
  65.   // Check if all static mappings have been met 
  66.   if ($this->_staticCount != $pathStaticCount) {//规则的所有普通字符串必须在path中得到匹配 
  67.    return false; 
  68.   } 
  69.   // 解析完后,规则定义的所有变量也必须全部出现,否则视为不匹配 
  70.   foreach ($this->_vars as $var) { 
  71.    if (!array_key_exists($var$return)) { 
  72.     return false; 
  73.    } 
  74.   } 
  75.   return $return
  76. }

Tags: Framework Zend框架路由

分享到: