无限级分类在开发中经常使用,例如:部门结构、文章分类。无限级分类的难点在于“输出”和“查询”,例如

  • 将文章分类输出为<ul>列表形式;
  • 查找分类A下面所有分类包含的文章。

1.实现原理

在《无限级分类实现思路》一文中介绍了几种常见的实现方法,各有利弊。其中“改进前序遍历树”数据结构,便于输出和查询,但是在移动分类和常规理解上有些复杂。

2.数据结构

id fid title
1 0 中国
2 1 江苏
3 1 安徽
4 8 江阴
5 3 芜湖
6 3 合肥
7 3 蚌埠
8 2 无锡
  1. <?php
  2. $list = array(
  3. array('id'=>1, 'fid'=>0, 'title' => '中国'),
  4. array('id'=>2, 'fid'=>1, 'title' => '江苏'),
  5. array('id'=>3, 'fid'=>1, 'title' => '安徽'),
  6. array('id'=>4, 'fid'=>8, 'title' => '江阴'),
  7. array('id'=>5, 'fid'=>3, 'title' => '芜湖'),
  8. array('id'=>6, 'fid'=>3, 'title' => '合肥'),
  9. array('id'=>7, 'fid'=>3, 'title' => '蚌埠'),
  10. array('id'=>8, 'fid'=>8, 'title' => '无锡')
  11. );
  12. ?>

各分类之间通过父类id(即fid)进行级别“串联”,形成一棵分类树。在进行串联时候有一点值得注意:分类A的fid不可以是其子类的id。

在使用这种数据结构进行输出时最常用的算法就是“递归”,熟悉PHP语言的朋友肯定知道,PHP不擅长递归 ,而且递归次数有限(100次左右,因操作系统和配置而异)。

由于所有的递归均可以使用循环实现,本文根据PHP语言特点编写了一套关于“无限级”分类的函数,相比递归实现而言效率更高

3.输出ul列表形式

将上述数据输出为下面的HTML

  1. <ul>
  2. <li class="first-child">
  3. <div>江苏</div>
  4. <ul>
  5. <li class="first-child last-child">
  6. <div>无锡</div>
  7. <ul>
  8. <li class="first-child last-child">
  9. <div>江阴</div>
  10. </li>
  11. </ul>
  12. </li>
  13. </ul>
  14. </li>
  15. <li class="last-child">
  16. <div>安徽</div>
  17. <ul>
  18. <li class="first-child"><div>芜湖</div></li>
  19. <li><div>合肥</div></li>
  20. <li class="last-child"><div>蚌埠</div></li>
  21. </ul>
  22. </li>
  23. </ul>

这种HTML结构在前端使用(使用JavaScript和CSS构造可折叠树)十分方便。具体实现程序如下:

  1. <ul><?php echo get_tree_ul($list, 1); ?></ul>

4.输出option列表形式

  1. <select>
  2. <option value="2">江苏</option>
  3. <option value="8">&nbsp;&nbsp;&nbsp;&nbsp;无锡</option>
  4. <option value="4">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;江阴</option>
  5. <option value="3">安徽</option>
  6. <option value="5">&nbsp;&nbsp;&nbsp;&nbsp;芜湖</option>
  7. <option value="6">&nbsp;&nbsp;&nbsp;&nbsp;合肥</option>
  8. <option value="7">&nbsp;&nbsp;&nbsp;&nbsp;蚌埠</option>
  9. </select>

具体实现程序如下:

  1. <select>
  2. <?php
  3. // get_tree_option()返回数组,并为每个元素增加了“深度”(即depth)列,直接输出即可
  4. $options = get_tree_option($list, 1);
  5. foreach($options as $op) {
  6. echo '<option value="' . $op['id'] .'">' . str_repeat(" ", $op['depth'] * 4) . $op['title'] . '<;/option>';
  7. }
  8. ?>
  9. <;/select>

5. 查找某一分类的所有子类

  1. <?php
  2. $children = get_tree_child($list, 0);
  3. echo implode(',', $children); // 输出:1,3,2,7,6,5,8,4
  4. ?>

6. 查找某一分类的所有父类

  1. <?php
  2. $children = get_tree_parent($list, 4);
  3. echo implode(',', $children); //8, 2, 10
  4. ?>

7. 相关函数

  1. <?php
  2. function get_tree_child($data, $fid) {
  3. $result = array();
  4. $fids = array($fid);
  5. do {
  6. $cids = array();
  7. $flag = false;
  8. foreach($fids as $fid) {
  9. for($i = count($data) - 1; $i >=0 ; $i--) {
  10. $node = $data[$i];
  11. if($node['fid'] == $fid) {
  12. array_splice($data, $i , 1);
  13. $result[] = $node['id'];
  14. $cids[] = $node['id'];
  15. $flag = true;
  16. }
  17. }
  18. }
  19. $fids = $cids;
  20. } while($flag === true);
  21. return $result;
  22. }
  23. function get_tree_parent($data, $id) {
  24. $result = array();
  25. $obj = array();
  26. foreach($data as $node) {
  27. $obj[$node['id']] = $node;
  28. }
  29. $value = isset($obj[$id]) ? $obj[$id] : null;
  30. while($value) {
  31. $id = null;
  32. foreach($data as $node) {
  33. if($node['id'] == $value['fid']) {
  34. $id = $node['id'];
  35. $result[] = $node['id'];
  36. break;
  37. }
  38. }
  39. if($id === null) {
  40. $result[] = $value['fid'];
  41. }
  42. $value = isset($obj[$id]) ? $obj[$id] : null;
  43. }
  44. unset($obj);
  45. return $result;
  46. }
  47. function get_tree_ul($data, $fid) {
  48. $stack = array($fid);
  49. $child = array();
  50. $added_left = array();
  51. $added_right= array();
  52. $html_left = array();
  53. $html_right = array();
  54. $obj = array();
  55. $loop = 0;
  56. foreach($data as $node) {
  57. $pid = $node['fid'];
  58. if(!isset($child[$pid])) {
  59. $child[$pid] = array();
  60. }
  61. array_push($child[$pid], $node['id']);
  62. $obj[$node['id']] = $node;
  63. }
  64. while (count($stack) > 0) {
  65. $id = $stack[0];
  66. $flag = false;
  67. $node = isset($obj[$id]) ? $obj[$id] : null;
  68. if (isset($child[$id])) {
  69. $cids = $child[$id];
  70. $length = count($cids);
  71. for($i = $length - 1; $i >= 0; $i--) {
  72. array_unshift($stack, $cids[$i]);
  73. }
  74. $obj[$cids[$length - 1]]['isLastChild'] = true;
  75. $obj[$cids[0]]['isFirstChild'] = true;
  76. $flag = true;
  77. }
  78. if ($id != $fid && $node && !isset($added_left[$id])) {
  79. if(isset($node['isFirstChild']) && isset($node['isLastChild'])) {
  80. $html_left[] = '<li class="first-child last-child">';
  81. } else if(isset($node['isFirstChild'])) {
  82. $html_left[] = '<li class="first-child">';
  83. } else if(isset($node['isLastChild'])) {
  84. $html_left[] = '<li class="last-child">';
  85. } else {
  86. $html_left[] = '<li>';
  87. }
  88. $html_left[] = ($flag === true) ? "<div>{$node['title']}</div><ul>" : "<div>{$node['title']}</div>";
  89. $added_left[$id] = true;
  90. }
  91. if ($id != $fid && $node && !isset($added_right[$id])) {
  92. $html_right[] = ($flag === true) ? '</ul></li>' : '</li>';
  93. $added_right[$id] = true;
  94. }
  95. if ($flag == false) {
  96. if($node) {
  97. $cids = $child[$node['fid']];
  98. for ($i = count($cids) - 1; $i >= 0; $i--) {
  99. if ($cids[$i] == $id) {
  100. array_splice($child[$node['fid']], $i, 1);
  101. break;
  102. }
  103. }
  104. if(count($child[$node['fid']]) == 0) {
  105. $child[$node['fid']] = null;
  106. }
  107. }
  108. array_push($html_left, array_pop($html_right));
  109. array_shift($stack);
  110. }
  111. $loop++;
  112. if($loop > 5000) return $html_left;
  113. }
  114. unset($child);
  115. unset($obj);
  116. return implode('', $html_left);
  117. }
  118. function get_tree_option($data, $fid) {
  119. $stack = array($fid);
  120. $child = array();
  121. $added = array();
  122. $options = array();
  123. $obj = array();
  124. $loop = 0;
  125. $depth = -1;
  126. foreach($data as $node) {
  127. $pid = $node['fid'];
  128. if(!isset($child[$pid])) {
  129. $child[$pid] = array();
  130. }
  131. array_push($child[$pid], $node['id']);
  132. $obj[$node['id']] = $node;
  133. }
  134. while (count($stack) > 0) {
  135. $id = $stack[0];
  136. $flag = false;
  137. $node = isset($obj[$id]) ? $obj[$id] : null;
  138. if (isset($child[$id])) {
  139. for($i = count($child[$id]) - 1; $i >= 0; $i--) {
  140. array_unshift($stack, $child[$id][$i]);
  141. }
  142. $flag = true;
  143. }
  144. if ($id != $fid && $node && !isset($added[$id])) {
  145. $node['depth'] = $depth;
  146. $options[] = $node;
  147. $added[$id] = true;
  148. }
  149. if($flag == true){
  150. $depth++;
  151. } else {
  152. if($node) {
  153. for ($i = count($child[$node['fid']]) - 1; $i >= 0; $i--) {
  154. if ($child[$node['fid']][$i] == $id) {
  155. array_splice($child[$node['fid']], $i, 1);
  156. break;
  157. }
  158. }
  159. if(count($child[$node['fid']]) == 0) {
  160. $child[$node['fid']] = null;
  161. $depth--;
  162. }
  163. }
  164. array_shift($stack);
  165. }
  166. $loop++;
  167. if($loop > 5000) return $options;
  168. }
  169. unset($child);
  170. unset($obj);
  171. return $options;
  172. }
  173. ?>

PHP无限级分类的实现(不使用递归)的更多相关文章

  1. (实用篇)PHP递归实现无限级分类

    在一些复杂的系统中,要求对信息栏目进行无限级的分类,以增强系统的灵活性.那么PHP是如何实现无限级分类的呢?我们在本文中使用递归算法并结合mysql数据表实现无限级分类. 在一些复杂的系统中,要求对信 ...

  2. PHP+MySQL无限级分类(非递归)

    要实现无限级分类,递归一般是第一个也是最容易想到的,但是递归一般被认为占用资源的方法,所以很多系统是不考虑使用递归的 本文还是通过数据库的设计,用一句sql语句实现 数据库字段大概如下: 字段 说明 ...

  3. PHP无限级分类-递归(不推荐)

    [http://www.helloweba.com/view-blog-204.html] 在一些复杂的系统中,要求对信息栏目进行无限级的分类,以增强系统的灵活性.那么PHP是如何实现无限级分类的呢? ...

  4. php非递归无限级分类.

    项目需要.递归无限级分类效率实在太低.理了半天思路写的. 分类越多效率越高. /** * 单次循环返回无限极分类嵌套 * @param array $data 操作的数组 * @param strin ...

  5. 转:php+mysql菜单无限级分类(非递归)

    php+mysql无限级分类(非递归) 参考:http://www.chhua.com/web-note3244

  6. PHP不使用递归的无限级分类

    不用递归实现无限级分类,简单测试了下性能比递归稍好一点点点,但写得太复杂了,还是递归简单方便点 代码: <?php $list = array( array('id'=>1, 'pid'= ...

  7. PHP迭代与递归实现无限级分类

    无限级分类是开发中常见的情况,因此本文对常见的无限极分类算法进行总结归纳. 1.循环迭代实现 $arr = [ 1=>['id'=>1,'name'=>'父1','father'=& ...

  8. C#无限级分类递归显示示例

    <%@ Page Language="C#" AutoEventWireup="true" CodeFile="RoleDemo20150305 ...

  9. PHP递归实现无限级分类

    在一些复杂的系统中,要求对信息栏目进行无限级的分类,以增强系统的灵活性.那么PHP是如何实现无限级分类的呢?我们在本文中使用递归算法并结合mysql数据表实现无限级分类. 在一些复杂的系统中,要求对信 ...

随机推荐

  1. 淘宝web前端开发岗面试经历及感悟

    今天下午四点接到淘宝UED的面试电话,很突然,很激动.现在怀着淡淡的忧伤为之文以志一下. 1.自我介绍一下. 我xx时候毕业,在xx公司任xx职务,主要负责xx balabala.(怕公司同事听到,接 ...

  2. GIS项目中数据开源、工具开源、开发开源的解决方案

    文章版权由作者李晓晖和博客园共有,若转载请于明显处标明出处:http://www.cnblogs.com/naaoveGIS/. 摆脱免费地图开发包的约束,拒绝商业地图软件的费用,高效.精确.完备是我 ...

  3. 关于JavaScript定时机制的总结

    要理解JavaScript的定时机制,就要知道JavaScript的运行机制. 首先声明,JavaScript是单线程运行(JavaScript引擎线程)事件驱动. 一.浏览器中有多个线程 一款浏览器 ...

  4. iframe在iphone6 plus的safari下子页面的宽度不受父页面控制的bug

    这是想要的效果: 样式设置是iframe外面的宽度为100%,iframe的宽度为父元素的90%,高度为宽度 除以1.6,固定比例, 正常显示就是上面的样子,但是,问题出现在iphone特定手机特定版 ...

  5. assign()

  6. Linux下查看版本号,查看存在的普通用户

    1. 查看版本号 uname -a ## 查看所有信息 uname --help ## 查看关于uname命令的帮助 2. 查看存在的普通用户 vim /etc/passwd ## 查看passwd文 ...

  7. php中防止SQL注入的方法

    [一.在服务器端配置] 安全,PHP代码编写是一方面,PHP的配置更是非常关键. 我们php手手工安装的,php的默认配置文件在 /usr/local/apache2/conf/php.ini,我们最 ...

  8. 复杂的xml转化为java实体

    一.样例一: 以根据订单号向支付宝查询支付是否成功为例(成功信息) 失败信息: <?xml version="1.0" encoding="utf-8"? ...

  9. SOA实践之基于服务总线的设计

    在上文中,主要介绍了SOA的概念,什么叫做“服务”,“服务”应该具备哪些特性.本篇中,我将介绍SOA的一种很常见的设计实践--基于服务总线的设计. 基于服务总线的设计 基于总线的设计,借鉴了计算机内部 ...

  10. 新闻类网站rss接口的编写心得

    使用的是Jdom中的相关API,具体步骤如下 要求的格式: <rss xmlns:content="http://purl.org/rss/1.0/modules/content/&q ...