0

0

无限级分类[预排序遍历树算法]

php中文网

php中文网

发布时间:2016-06-08 17:29:47

|

1253人浏览过

|

来源于php中文网

原创

无限级分类[预排序遍历树算法]

预排序遍历树算法

现在让我们看一看另外一种不使用递归计算,更加快速的方法,这就是预排序遍历树算法(modified preorder tree traversal algorithm)
这种方法大家可能接触的比较少,初次使用也不像上面的方法容易理解,但是由于这种方法不使用递归查询算法,有更高的查询效率。

我们首先将多级数据按照下面的方式画在纸上,在根节点Food的左侧写上 1 然后沿着这个树继续向下 在 Fruit 的左侧写上 2 然后继续前进,沿着整个树的边缘给每一个节点都标上左侧和右侧的数字。最后一个数字是标在Food 右侧的 18。 在下面的这张图中你可以看到整个标好了数字的多级结构。(没有看懂?用你的手指指着数字从1数到18就明白怎么回事了。还不明白,再数一遍,注意移动你的手指)。
这些数字标明了各个节点之间的关系,"Red"的号是3和6,它是 "Food" 1-18 的子孙节点。 同样,我们可以看到 所有左值大于2和右值小于11的节点 都是"Fruit" 2-11 的子孙节点

以下是代码:                          1 Food 18

                             |

            +------------------------------+

            |                              |

        2 Fruit 11                     12 Meat 17

            |                              |

    +-------------+                 +------------+

    |             |                 |            |

3 Red 6      7 Yellow 10       13 Beef 14   15 Pork 16

    |             |

4 Cherry 5    8 Banana 9
复制代码这样整个树状结构可以通过左右值来存储到数据库中。继续之前,我们看一看下面整理过的数据表。

以下是代码: +----------+------------+-----+-----+

|  parent  |    name    | lft | rgt |

+----------+------------+-----+-----+

|          |    Food    | 1   | 18  |

|   Food   |   Fruit    | 2   | 11  |

|   Fruit  |    Red     | 3   |  6  |

|   Red    |    Cherry  | 4   |  5  |

|   Fruit  |    Yellow  | 7   | 10  |

|   Yellow |    Banana  | 8   |  9  |

|   Food   |    Meat    | 12  | 17  |

|   Meat   |    Beef    | 13  | 14  |

|   Meat   |    Pork    | 15  | 16  |

+----------+------------+-----+-----+
复制代码注意:由于"left"和"right"在 SQL中有特殊的意义,所以我们需要用"lft"和"rgt"来表示左右字段。 另外这种结构中不再需要"parent"字段来表示树状结构。也就是 说下面这样的表结构就足够了。

以下是代码: +------------+-----+-----+

|    name    | lft | rgt |

+------------+-----+-----+

|    Food    | 1   | 18  |

|    Fruit   | 2   | 11  |

|    Red     | 3   |  6  |

|    Cherry  | 4   |  5  |

|    Yellow  | 7   | 10  |

|    Banana  | 8   |  9  |

|    Meat    | 12  | 17  |

|    Beef    | 13  | 14  |

|    Pork    | 15  | 16  |

+------------+-----+-----+
复制代码好了我们现在可以从数据库中获取数据了,例如我们需要得到"Fruit"项下的所有所有节点就可以这样写查询语句: SELECT * FROM tree WHERE lft BETWEEN 2 AND 11;
复制代码这个查询得到了以下的结果。

以下是代码: +------------+-----+-----+

|    name    | lft | rgt |

+------------+-----+-----+

|    Fruit   | 2   | 11  |

|    Red     | 3   |  6  |

|    Cherry  | 4   |  5  |

|    Yellow  | 7   | 10  |

|    Banana  | 8   |  9  |

+------------+-----+-----+
复制代码看到了吧,只要一个查询就可以得到所有这些节点。为了能够像上面的递归函数那样显示整个树状结构,我们还需要对这样的查询进行排序。用节点的左值进行排序: SELECT * FROM tree WHERE lft BETWEEN 2 AND 11 ORDER BY lft ASC;
复制代码剩下的问题如何显示层级的缩进了。
以下是代码:

function display_tree($root) {

    // 得到根节点的左右值

    $result = mysql_query("

        SELECT lft, rgt

        FROM tree

Reecho睿声
Reecho睿声

Reecho AI:超拟真语音合成与瞬时语音克隆平台

下载

        WHERE name = '" . $root . "'

        ;"

    );

    $row = mysql_fetch_array($result);

    // 准备一个空的右值堆栈

    $right = array();

    // 获得根基点的所有子孙节点

    $result = mysql_query("

        SELECT name, lft, rgt

        FROM tree

        WHERE lft BETWEEN '" . $row['lft'] . "' AND '" . $row['rgt'] ."'

        ORDER BY lft ASC

        ;"

    );

    // 显示每一行

    while ($row = mysql_fetch_array($result)) {

        // only check stack if there is one

        if (count($right) > 0) {

            // 检查我们是否应该将节点移出堆栈

            while ($right[count($right) - 1]

                array_pop($right);

            }

        }

        // 缩进显示节点的名称

        echo str_repeat('  ',count($right)) . $row['name'] . "n";

        // 将这个节点加入到堆栈中

        $right[] = $row['rgt'];

    }

}

?>


复制代码如果你运行一下以上的函数就会得到和递归函数一样的结果。只是我们的这个新的函数可能会更快一些,因为只有2次数据库查询。
要获知一个节点的路径就更简单了,如果我们想知道Cherry 的路径就利用它的左右值4和5来做一个查询。 SELECT name FROM tree WHERE lft ; 5 ORDER BY lft ASC;
复制代码这样就会得到以下的结果:

以下是代码: +------------+

|    name    |

+------------+

|    Food    |

|    Fruit   |

|    Red     |

+------------+
复制代码那么某个节点到底有多少子孙节点呢?很简单,子孙总数=(右值-左值-1)/2 descendants = (right – left - 1) / 2
复制代码不相信?自己算一算啦。
用这个简单的公式,我们可以很快的算出"Fruit 2-11"节点有4个子孙节点,而"Banana 8-9"节点没有子孙节点,也就是说它不是一个父节点了。
很神奇吧?虽然我已经多次用过这个方法,但是每次这样做的时候还是感到很神奇。
这的确是个很好的办法,但是有什么办法能够帮我们建立这样有左右值的数据表呢?这里再介绍一个函数给大家,这个函数可以将name和parent结构的表自动转换成带有左右值的数据表。
以下是代码:

function rebuild_tree($parent, $left) {

    // the right value of this node is the left value + 1

    $right = $left+1;

    // get all children of this node

    $result = mysql_query("

        SELECT name

        FROM tree

        WHERE parent = '" . $parent . "'

        ;"

    );

    while ($row = mysql_fetch_array($result)) {

        // recursive execution of this function for each

        // child of this node

        // $right is the current right value, which is

        // incremented by the rebuild_tree function

        $right = rebuild_tree($row['name'], $right);

    }

    // we've got the left value, and now that we've processed

    // the children of this node we also know the right value

    mysql_query("

        UPDATE tree

        SET

            lft = '" . $left . "',

            rgt= '" . $right . "'

        WHERE name = '" . $parent . "'

        ;"

    );

    // return the right value of this node + 1

    return $right + 1;

}

?>


复制代码当然这个函数是一个递归函数,我们需要从根节点开始运行这个函数来重建一个带有左右值的树 rebuild_tree('Food',1);


复制代码这个函数看上去有些复杂,但是它的作用和手工对表进行编号一样,就是将立体多层结构的转换成一个带有左右值的数据表。

那么对于这样的结构我们该如何增加,更新和删除一个节点呢?
增加一个节点一般有两种方法:
第一种,保留原有的name 和parent结构,用老方法向数据中添加数据,每增加一条数据以后使用rebuild_tree函数对整个结构重新进行一次编号。
第二种,效率更高的办法是改变所有位于新节点右侧的数值。举例来说:我们想增加一种新的水果"Strawberry"(草莓)它将成为"Red"节点的最后一个子节点。首先我们需要为它腾出一些空间。"Red"的右值应当从6改成8,"Yellow 7-10 "的左右值则应当改成 9-12。 依次类推我们可以得知,如果要给新的值腾出空间需要给所有左右值大于5的节点 (5 是"Red"最后一个子节点的右值) 加上2。 所以我们这样进行数据库操作: UPDATE tree SET rgt = rgt + 2 WHERE rgt > 5;

UPDATE tree SET lft = lft + 2 WHERE lft > 5;
复制代码这样就为新插入的值腾出了空间,现在可以在腾出的空间里建立一个新的数据节点了, 它的左右值分别是6和7 INSERT INTO tree SET lft=6, rgt=7, name='Strawberry';
复制代码再做一次查询看看吧!怎么样?很快吧。

四、结语
好了,现在你可以用两种不同的方法设计你的多级数据库结构了,采用何种方式完全取决于你个人的判断,但是对于层次多数量大的结构我更喜欢第二种方法。如果查询量较小但是需要频繁添加和更新的数据,则第一种方法更为简便。
另外,如果数据库支持的话 你还可以将rebuild_tree()和 腾出空间的操作写成数据库端的触发器函数, 在插入和更新的时候自动执行, 这样可以得到更好的运行效率, 而且你添加新节点的SQL语句会变得更加简单。

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
go语言 注释编码
go语言 注释编码

本专题整合了go语言注释、注释规范等等内容,阅读专题下面的文章了解更多详细内容。

59

2026.01.31

go语言 math包
go语言 math包

本专题整合了go语言math包相关内容,阅读专题下面的文章了解更多详细内容。

52

2026.01.31

go语言输入函数
go语言输入函数

本专题整合了go语言输入相关教程内容,阅读专题下面的文章了解更多详细内容。

25

2026.01.31

golang 循环遍历
golang 循环遍历

本专题整合了golang循环遍历相关教程,阅读专题下面的文章了解更多详细内容。

10

2026.01.31

Golang人工智能合集
Golang人工智能合集

本专题整合了Golang人工智能相关内容,阅读专题下面的文章了解更多详细内容。

7

2026.01.31

2026赚钱平台入口大全
2026赚钱平台入口大全

2026年最新赚钱平台入口汇总,涵盖任务众包、内容创作、电商运营、技能变现等多类正规渠道,助你轻松开启副业增收之路。阅读专题下面的文章了解更多详细内容。

411

2026.01.31

高干文在线阅读网站大全
高干文在线阅读网站大全

汇集热门1v1高干文免费阅读资源,涵盖都市言情、京味大院、军旅高干等经典题材,情节紧凑、人物鲜明。阅读专题下面的文章了解更多详细内容。

232

2026.01.31

无需付费的漫画app大全
无需付费的漫画app大全

想找真正免费又无套路的漫画App?本合集精选多款永久免费、资源丰富、无广告干扰的优质漫画应用,涵盖国漫、日漫、韩漫及经典老番,满足各类阅读需求。阅读专题下面的文章了解更多详细内容。

194

2026.01.31

漫画免费在线观看地址大全
漫画免费在线观看地址大全

想找免费又资源丰富的漫画网站?本合集精选2025-2026年热门平台,涵盖国漫、日漫、韩漫等多类型作品,支持高清流畅阅读与离线缓存。阅读专题下面的文章了解更多详细内容。

150

2026.01.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Excel 教程
Excel 教程

共162课时 | 15万人学习

Pandas 教程
Pandas 教程

共15课时 | 1万人学习

C# 教程
C# 教程

共94课时 | 8.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号