PHP实现无限极分类的两种方式示例【递归和引用方式】


Posted in PHP onMarch 25, 2019

本文实例讲述了PHP实现无限极分类的两种方式。分享给大家供大家参考,具体如下:
面试的时候被问到无限极分类的设计和实现,比较常见的做法是在建表的时候,增加一个PID字段用来区别自己所属的分类

$array = array(
array('id' => 1, 'pid' => 0, 'name' => '河北省'),
array('id' => 2, 'pid' => 0, 'name' => '北京市'),
array('id' => 3, 'pid' => 1, 'name' => '邯郸市'),
array('id' => 4, 'pid' => 2, 'name' => '朝阳区'),
array('id' => 5, 'pid' => 2, 'name' => '通州区'),
array('id' => 6, 'pid' => 4, 'name' => '望京'),
array('id' => 7, 'pid' => 4, 'name' => '酒仙桥'),
array('id' => 8, 'pid' => 3, 'name' => '永年区'),
array('id' => 9, 'pid' => 1, 'name' => '武安市'),
);

数据在数据库中存储大概是这个样子,怎么实现无限极递归呢,有两种常用的做法,递归和引用算法

递归算法

/**
* 递归实现无限极分类
* @param $array 分类数据
* @param $pid 父ID
* @param $level 分类级别
* @return $list 分好类的数组 直接遍历即可 $level可以用来遍历缩进
*/
function getTree($array, $pid =0, $level = 0){
    //声明静态数组,避免递归调用时,多次声明导致数组覆盖
    static $list = [];
    foreach ($array as $key => $value){
      //第一次遍历,找到父节点为根节点的节点 也就是pid=0的节点
      if ($value['pid'] == $pid){
        //父节点为根节点的节点,级别为0,也就是第一级
        $value['level'] = $level;
        //把数组放到list中
        $list[] = $value;
        //把这个节点从数组中移除,减少后续递归消耗
        unset($array[$key]);
        //开始递归,查找父ID为该节点ID的节点,级别则为原级别+1
        getTree($array, $value['id'], $level+1);
      }
    }
    return $list;
}
/*
* 获得递归完的数据,遍历生成分类
*/
$array = getTree($array);
foreach($array) as $value{
    echo str_repeat('--', $value['level']), $value['name'].'<br />';
}

输出结果 无限极分类实现ok

河北省
--邯郸市
----永年区
--武安市
北京市
--朝阳区
----望京
----酒仙桥
--通州区

引用算法

function generateTree($array){
  //第一步 构造数据
  $items = array();
  foreach($array as $value){
    $items[$value['id']] = $value;
  }
  //第二部 遍历数据 生成树状结构
  $tree = array();
  foreach($items as $key => $value){
    if(isset($items[$item['pid']])){
      $items[$item['pid']]['son'][] = &$items[$key];
    }else{
      $tree[] = &$items[$key];
    }
  }
  return $tree;
}
//经过第一步 数据变成了这样
Array
(
  [1] => Array
    (
      [id] => 1
      [pid] => 0
      [name] => 河北省
      [children] => Array
        (
        )
    )
  [2] => Array
    (
      [id] => 2
      [pid] => 0
      [name] => 北京市
      [children] => Array
        (
        )
    )
  [3] => Array
    (
      [id] => 3
      [pid] => 1
      [name] => 邯郸市
      [children] => Array
        (
        )
    )
  [4] => Array
    (
      [id] => 4
      [pid] => 2
      [name] => 朝阳区
      [children] => Array
        (
        )
    )
  [5] => Array
    (
      [id] => 5
      [pid] => 2
      [name] => 通州区
      [children] => Array
        (
        )
    )
  [6] => Array
    (
      [id] => 6
      [pid] => 4
      [name] => 望京
      [children] => Array
        (
        )
    )
  [7] => Array
    (
      [id] => 7
      [pid] => 4
      [name] => 酒仙桥
      [children] => Array
        (
        )
    )
  [8] => Array
    (
      [id] => 8
      [pid] => 3
      [name] => 永年区
      [children] => Array
        (
        )
    )
  [9] => Array
    (
      [id] => 9
      [pid] => 1
      [name] => 武安市
      [children] => Array
        (
        )
    )
)
//第一步很容易就能看懂,就是构造数据,现在咱们仔细说一下第二步
 $tree = array();
 //遍历构造的数据
  foreach($items as $key => $value){
  //如果pid这个节点存在
    if(isset($items[$value['pid']])){
      //把当前的$value放到pid节点的son中 注意 这里传递的是引用 为什么呢?
      $items[$value['pid']]['son'][] = &$items[$key];
    }else{
      $tree[] = &$items[$key];
    }
  }
//这个方法的核心在于引用,php变量默认的传值方式是按指传递
//也就是说 假如说 遍历顺序是 河北省 邯郸市 当遍历到河北省时 会把河北省放到tree中 遍历到邯郸市时 会把邯郸市放到河北省的子节点数组中 但是!!! 这会儿的tree数组中 河北省已经放进去了 根据php变量按值传递的规则 你并没有更改tree数组中的河北省的数据 所以这里用到了引用传递
//当你对河北省做更改时,tree数组中的河北省也一并做了更改 下面我们做个实验 我们把引用传递去掉,看一下结果
//使用普通传值输出结果
 Array
(
  [0] => Array
    (
      [id] => 1
      [pid] => 0
      [name] => 河北省
    )
  [1] => Array
    (
      [id] => 2
      [pid] => 0
      [name] => 北京市
    )
)
//可以看到 只有河北省和北京市输出出来了 因为他们俩是第一级节点 而且排行1和2,放到$tree数组中之后,没有使用引用传递,那么后续对他俩的子节点的操作都没有在$tree中生效,现在我们更改一下顺序 把邯郸市放到河北省的前面 那么根据咱们的推断 那么邯郸市就应该出现在tree数组里
//邯郸市放到河北省前面的输出结果
Array
(
  [0] => Array
    (
      [id] => 1
      [pid] => 0
      [name] => 河北省
      [son] => Array
        (
          [0] => Array
            (
              [id] => 3
              [pid] => 1
              [name] => 邯郸市
            )
        )
    )
  [1] => Array
    (
      [id] => 2
      [pid] => 0
      [name] => 北京市
    )
)
//果然是这样 那么证明我们的推断是正确的 现在我们把引用传值改回去 再看一下
//使用引用传值输出结果
Array
(
  [1] => Array
    (
      [id] => 1
      [pid] => 0
      [name] => 河北省
      [children] => Array
        (
          [0] => Array
            (
              [id] => 3
              [pid] => 1
              [name] => 邯郸市
              [children] => Array
                (
                  [0] => Array
                    (
                      [id] => 8
                      [pid] => 3
                      [name] => 永年区
                    )
                )
            )
          [1] => Array
            (
              [id] => 9
              [pid] => 1
              [name] => 武安市
            )
        )
    )
  [2] => Array
    (
      [id] => 2
      [pid] => 0
      [name] => 北京市
      [children] => Array
        (
          [0] => Array
            (
              [id] => 4
              [pid] => 2
              [name] => 朝阳区
              [children] => Array
                (
                  [0] => Array
                    (
                      [id] => 6
                      [pid] => 4
                      [name] => 望京
                    )
                  [1] => Array
                    (
                      [id] => 7
                      [pid] => 4
                      [name] => 酒仙桥
                    )
                )
            )
          [1] => Array
            (
              [id] => 5
              [pid] => 2
              [name] => 通州区
            )
        )
    )
)
//树状结构完美的输出出来了 这个方法的核心就是引用传值

希望本文所述对大家PHP程序设计有所帮助。

PHP 相关文章推荐
PHP+MySQL5.0中文乱码解决方法
Nov 20 PHP
php定时计划任务与fsockopen持续进程实例
May 23 PHP
通过php修改xml文档内容的方法
Jan 23 PHP
PHP实现抓取Google IP并自动修改hosts文件
Feb 12 PHP
Thinkphp实现自动验证和自动完成
Dec 19 PHP
php将一维数组转换为每3个连续值组成的二维数组
May 06 PHP
php使用文本统计访问量的方法
May 12 PHP
PHP三种方式实现链式操作详解
Jan 21 PHP
php实现mysql连接池效果实现代码
Jan 25 PHP
PHP实现网站应用微信登录功能详解
Apr 11 PHP
Smarty模板变量与调节器实例详解
Jul 20 PHP
设定php简写功能的方法
Nov 28 PHP
详解PHP神奇又有用的Trait
Mar 25 #PHP
PHP自动载入类文件函数__autoload的使用方法
Mar 25 #PHP
PHP生成短网址的思路以及实现方法的详解
Mar 25 #PHP
PHP错误提示It is not safe to rely on the system……的解决方法
Mar 25 #PHP
mongodb和php的用法详解
Mar 25 #PHP
PHP随机数函数rand()与mt_rand()的讲解
Mar 25 #PHP
php微信扫码支付 php公众号支付
Mar 24 #PHP
You might like
php笔记之:php数组相关函数的使用
2013/04/26 PHP
PHP邮件群发机实现代码
2016/02/16 PHP
javascript parseInt 大改造
2009/09/27 Javascript
Javascript中的相等与不等运算
2010/04/25 Javascript
深入理解JavaScript系列(6):S.O.L.I.D五大原则之单一职责SRP
2012/01/15 Javascript
js模仿html5 placeholder适应于不支持的浏览器
2013/01/13 Javascript
解析Javascript中难以理解的11个问题
2013/12/09 Javascript
js实现简单选项卡与自动切换效果的方法
2015/04/10 Javascript
jquery插件splitScren实现页面分屏切换模板特效
2015/06/16 Javascript
jQuery事件绑定用法详解(附bind和live的区别)
2016/01/19 Javascript
为什么JavaScript没有块级作用域
2016/05/22 Javascript
JavaScript中style.left与offsetLeft的使用及区别详解
2016/06/08 Javascript
基于JS实现导航条之调用网页助手小精灵的方法
2016/06/17 Javascript
AngularJS通过$location获取及改变当前页面的URL
2016/09/23 Javascript
用Angular实时获取本地Localstorage数据,实现一个模拟后台数据登入的效果
2016/11/09 Javascript
Vue-Router的使用方法
2018/09/05 Javascript
vue 组件开发原理与实现方法详解
2019/11/29 Javascript
[03:42]2016国际邀请赛中国区预选赛首日现场玩家采访
2016/06/26 DOTA
[02:37]TI8勇士令状不朽珍藏II视频展示
2018/06/23 DOTA
python自定义异常实例详解
2017/07/11 Python
Python编程之序列操作实例详解
2017/07/22 Python
轻松实现TensorFlow微信跳一跳的AI
2018/01/05 Python
Python设计模式之建造者模式实例详解
2019/01/17 Python
Python PyInstaller安装和使用教程详解
2020/01/08 Python
Python文件操作方法详解
2020/02/09 Python
如何理解python面向对象编程
2020/06/01 Python
伦敦著名的运动鞋综合商店:Footpatrol
2019/03/25 全球购物
英国最大的户外商店:Go Outdoors
2019/04/17 全球购物
物流毕业生个人的自我评价
2014/02/13 职场文书
关于迟到的检讨书
2015/05/06 职场文书
奥巴马开学演讲观后感
2015/06/12 职场文书
小学运动会入场口号
2015/12/24 职场文书
银行岗位培训心得体会
2016/01/09 职场文书
2016年学生会感恩节活动总结
2016/04/01 职场文书
用python修改excel表某一列内容的操作方法
2021/06/11 Python
Anaconda安装pytorch和paddle的方法步骤
2022/04/03 Python