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

2025-05-29 0 65

本文分析了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

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25
public function __construct($route, $defaults = array(), $reqs = array())

{

$route = trim($route, $this->_urlDelimiter); //去掉规则首尾的url分隔符(默认是/)

$this->_defaults = (array) $defaults; //默认值数组,以变量名为键

$this->_requirements = (array) $reqs; //变量需要满足的正则表达式,以变量名为键

if ($route != '') {

foreach (explode($this->_urlDelimiter, $route) as $pos => $part) {

//把规则切分为一个数组

if (substr($part, 0, 1) == $this->_urlVariable) {//如果是一个变量的定义

$name = substr($part, 1); //获取变量名

//如果该变量定义了对应的正则表达式,则获取该表达式,否则置为null

$regex = (isset($reqs[$name]) ? $reqs[$name] : $this->_defaultRegex);

//_parts数组包含了规则的各个部分,如果是变量的话,数组中有name元素

$this->_parts[$pos] = array('name' => $name, 'regex' => $regex);

//_vars包含了该规则中的所有变量的名字

$this->_vars[] = $name;

} else { //普通字符串

$this->_parts[$pos] = array('regex' => $part);

if ($part != '*') {

$this->_staticCount++; //该规则的普通字符串的个数

}

}

}

}

}

2、匹配算法

?

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

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

65

66

67

68

69

70

71

72

73

74

75

76

77
public function match($path)

{

$pathStaticCount = 0;

$defaults = $this->_defaults; //默认值数组,数组元素的键值是变量名

//默认值数组的一个拷贝,不过变量的值全部换成布尔值,其实这个值并没有实际用处,下面程序仅仅

//是通过判断键值是否存在而确定是否包含一个变量,可能这么做是为了节省空间,不过要是这样的话

//不如直接使用 $this->_defaults了?

if (count($defaults)) {

$unique = array_combine(array_keys($defaults), array_fill(0, count($defaults), true));

} else {

$unique = array();

}

$path = trim($path, $this->_urlDelimiter); //传入的path是已经去掉baseUrl的,这里确保去掉首尾的分隔符

if ($path != '') {

$path = explode($this->_urlDelimiter, $path);

foreach ($path as $pos => $pathPart) {

if (!isset($this->_parts[$pos])) {

//把path根据url分隔符分割为数组后,把每一部分和规则的对应部分比较,如果path中存在,

//而规则中不存在对应部分,那么该规则肯定不匹配,这里要注意$pos,是通过它把规则

//和path的对应部分对应起来。

return false;

}

if ($this->_parts[$pos]['regex'] == '*') {

//如果规则的当前部分是通配符*,则把path的剩余部分解释为url传递过来的变量,他们按照

//“变量名/变量值”这样的形式成对出现

$parts = array_slice($path, $pos); //获取path的剩余部分

$this->_getWildcardData($parts, $unique);

break;

}

$part = $this->_parts[$pos];

$name = isset($part['name']) ? $part['name'] : null;

$pathPart = urldecode($pathPart);//对传过来的值进行解码

if ($name === null) {//普通字符串,和规则的对应部分比较是否相等即可

if ($part['regex'] != $pathPart) {

return false;

}

} elseif ($part['regex'] === null) {

//如果是变量,但是没有需要满足的正则表达式,那么只有值不为空就可以了

if (strlen($pathPart) == 0) {

return false;

}

} else {//如果对该变量需要满足一个正则表达式,那么这里进行验证

$regex = $this->_regexDelimiter . '^' . $part['regex'] . '$' . $this->_regexDelimiter . 'iu';

if (!preg_match($regex, $pathPart)) {

return false;

}

}

if ($name !== null) {

// 如果是一个变量,则设置变量的值

$this->_values[$name] = $pathPart;

$unique[$name] = true; //其实没有必要设置,这个版本根本就没有用它

} else {

//把普通字符串的匹配计数加1,因为规则中的普通字符串是必须在path中存在的,否则就是

//匹配失败

$pathStaticCount++;

}

}

}

//$this->_values中保存的是分析获取的变量,如果规则中存在‘*',则$this->_params是获取的

//变量,否则是空数组,$this->_defaults是规则提供的默认变量值,这里用‘+'把三个数组相加

//这样的好处是如果后面的数组与前面的数组有相同的非整数的键值,后面的不会覆盖前面的,这

//与array_merge函数有区别,后者是会覆盖的。也就是说,如果$this->_values 中已经有键controller

//,那么$this->_defaults中的controller元素就被忽略,这样就$this->_defaults中的默认值只有在path

//中不存在的时候才会出现在返回值中。

$return = $this->_values + $this->_params + $this->_defaults;

// Check if all static mappings have been met

if ($this->_staticCount != $pathStaticCount) {//规则的所有普通字符串必须在path中得到匹配

return false;

}

// 解析完后,规则定义的所有变量也必须全部出现,否则视为不匹配

foreach ($this->_vars as $var) {

if (!array_key_exists($var, $return)) {

return false;

}

}

return $return;

}

希望本文所述对大家基于Zend Framework框架的PHP程序设计有所帮助。

收藏 (0) 打赏

感谢您的支持,我会继续努力的!

打开微信/支付宝扫一扫,即可进行扫码打赏哦,分享从这里开始,精彩与您同在
点赞 (0)

声明:本站所有文章,如无特殊说明或标注,均为本站原创发布。任何个人或组织,在未征得本站同意时,禁止复制、盗用、采集、发布本站内容到任何网站、书籍等各类媒体平台。如若本站内容侵犯了原著者的合法权益,可联系我们进行处理。

快网idc优惠网 建站教程 Zend Framework框架路由机制代码分析 https://www.kuaiidc.com/98531.html

相关文章

发表评论
暂无评论