我使用邻接表方法在 MySQL 数据库中存储一棵树
当我想预览树时,PHP 从数据库中检索整个树并使用递归进行预览。
但是迭代在性能上比递归要好,所以我想使用 for 循环填充树以获得更好的性能。
我不想使用任何 MySQL 函数、方法或触发器,我只想通过使用迭代(for 循环)来填充树
我使用邻接表方法在 MySQL 数据库中存储一棵树
当我想预览树时,PHP 从数据库中检索整个树并使用递归进行预览。
但是迭代在性能上比递归要好,所以我想使用 for 循环填充树以获得更好的性能。
我不想使用任何 MySQL 函数、方法或触发器,我只想通过使用迭代(for 循环)来填充树
在寻找帮助回答 IRC 上类似问题的链接时,我偶然发现了这个未回答的老问题。
这是一个粗略的示例,展示了如何从邻接列表构建路径列表,它要求数据按“父”和“id”排序,这可以很容易地转换为SELECT
带有ORDER BY parent, id
子句的 SQL:
<?php
$arr = [
['id' => 1, 'parent' => 0],
['id' => 2, 'parent' => 1],
['id' => 3, 'parent' => 2],
['id' => 4, 'parent' => 3],
['id' => 5, 'parent' => 0],
['id' => 6, 'parent' => 5],
['id' => 7, 'parent' => 5],
['id' => 8, 'parent' => 7],
['id' => 9, 'parent' => 8],
['id' => 10, 'parent' => 5],
['id' => 11, 'parent' => 0],
['id' => 12, 'parent' => 5],
['id' => 13, 'parent' => 5],
['id' => 14, 'parent' => 11],
['id' => 15, 'parent' => 13],
['id' => 16, 'parent' => 1],
['id' => 17, 'parent' => 15],
];
usort($arr, function($a, $b) { return $a['parent'] - $b['parent']?: $a['id'] - $b['id']; });
$paths = [];
foreach($arr as $current) {
if($current['parent'] > 0) {
$paths[ $current['id'] ] = array_merge($paths[ $current['parent'] ], [ $current['parent'] ]);
} else {
$paths[ $current['id'] ] = [];
}
}
ksort($paths);
foreach($paths as $k => $v) {
printf("%s => %s\n", $k, implode('/', array_merge($v, [ $k ])));
}
输出:
1 => 1
2 => 1/2
3 => 1/2/3
4 => 1/2/3/4
5 => 5
6 => 5/6
7 => 5/7
8 => 5/7/8
9 => 5/7/8/9
10 => 5/10
11 => 11
12 => 5/12
13 => 5/13
14 => 11/14
15 => 5/13/15
16 => 1/16
17 => 5/13/15/17