Joh*_*ohn 34 php arrays tree recursion multidimensional-array
我正在尝试创建一个包含任意数量子类别的类别列表,其中子类别也可以有自己的子类别.
我从Mysql数据库中选择了所有类别,猫在标准关联数组列表中,每个类别都有一个id,name,parentid,如果它是最高级别,则parentid为0.
我基本上希望能够获取单级别的猫数组并将其转换为多维数组结构,其中每个类别都可以包含一个包含子数组的元素.
现在,我可以通过循环查询每个类别来轻松实现这一点,但这远非理想,我试图在数据库上没有任何额外的命中.
我知道我需要一个递归函数.任何人都可以指出我正确的方向为这种树型结构?
干杯
Arn*_*anc 79
这样做的工作:
$items = array(
(object) array('id' => 42, 'parent_id' => 1),
(object) array('id' => 43, 'parent_id' => 42),
(object) array('id' => 1, 'parent_id' => 0),
);
$childs = array();
foreach($items as $item)
$childs[$item->parent_id][] = $item;
foreach($items as $item) if (isset($childs[$item->id]))
$item->childs = $childs[$item->id];
$tree = $childs[0];
print_r($tree);
Run Code Online (Sandbox Code Playgroud)
这首先通过parent_id索引类别.然后为每个类别中,我们只需要设置category->childs
到childs[category->id]
,并且树建!
所以,现在$tree
是类别树.它包含一个parent_id = 0的项目数组,它们本身包含一个子数组,它们本身就是......
产量print_r($tree)
:
stdClass Object
(
[id] => 1
[parent_id] => 0
[childs] => Array
(
[0] => stdClass Object
(
[id] => 42
[parent_id] => 1
[childs] => Array
(
[0] => stdClass Object
(
[id] => 43
[parent_id] => 42
)
)
)
)
)
Run Code Online (Sandbox Code Playgroud)
所以这是最后的功能:
function buildTree($items) {
$childs = array();
foreach($items as $item)
$childs[$item->parent_id][] = $item;
foreach($items as $item) if (isset($childs[$item->id]))
$item->childs = $childs[$item->id];
return $childs[0];
}
$tree = buildTree($items);
Run Code Online (Sandbox Code Playgroud)
$items = array(
array('id' => 42, 'parent_id' => 1),
array('id' => 43, 'parent_id' => 42),
array('id' => 1, 'parent_id' => 0),
);
$childs = array();
foreach($items as &$item) $childs[$item['parent_id']][] = &$item;
unset($item);
foreach($items as &$item) if (isset($childs[$item['id']]))
$item['childs'] = $childs[$item['id']];
unset($item);
$tree = $childs[0];
Run Code Online (Sandbox Code Playgroud)
所以最终函数的数组版本:
function buildTree($items) {
$childs = array();
foreach($items as &$item) $childs[$item['parent_id']][] = &$item;
unset($item);
foreach($items as &$item) if (isset($childs[$item['id']]))
$item['childs'] = $childs[$item['id']];
return $childs[0];
}
$tree = buildTree($items);
Run Code Online (Sandbox Code Playgroud)
Tha*_*hai 18
您可以一次获取所有类别.
假设您从数据库得到一个平坦的结果,如下所示:
$categories = array(
array('id' => 1, 'parent' => 0, 'name' => 'Category A'),
array('id' => 2, 'parent' => 0, 'name' => 'Category B'),
array('id' => 3, 'parent' => 0, 'name' => 'Category C'),
array('id' => 4, 'parent' => 0, 'name' => 'Category D'),
array('id' => 5, 'parent' => 0, 'name' => 'Category E'),
array('id' => 6, 'parent' => 2, 'name' => 'Subcategory F'),
array('id' => 7, 'parent' => 2, 'name' => 'Subcategory G'),
array('id' => 8, 'parent' => 3, 'name' => 'Subcategory H'),
array('id' => 9, 'parent' => 4, 'name' => 'Subcategory I'),
array('id' => 10, 'parent' => 9, 'name' => 'Subcategory J'),
);
Run Code Online (Sandbox Code Playgroud)
您可以创建一个简单的函数,将该平面列表转换为结构,最好是在函数内部.我使用pass-by-reference,因此每个类别只有一个数组,而不是一个类别的数组的多个副本.
function categoriesToTree(&$categories) {
Run Code Online (Sandbox Code Playgroud)
地图用于快速查找类别.在这里,我还为"root"级别创建了一个虚拟数组.
$map = array(
0 => array('subcategories' => array())
);
Run Code Online (Sandbox Code Playgroud)
我在每个类别数组中添加了另一个字段子类别,并将其添加到地图中.
foreach ($categories as &$category) {
$category['subcategories'] = array();
$map[$category['id']] = &$category;
}
Run Code Online (Sandbox Code Playgroud)
再次循环遍历每个类别,将其自身添加到其父类的子类别列表中.此处的引用很重要,否则当有更多子类别时,已添加的类别将不会更新.
foreach ($categories as &$category) {
$map[$category['parent']]['subcategories'][] = &$category;
}
Run Code Online (Sandbox Code Playgroud)
最后,返回指向所有顶级类别的虚拟类别的子类别._
return $map[0]['subcategories'];
}
Run Code Online (Sandbox Code Playgroud)
用法:
$tree = categoriesToTree($categories);
Run Code Online (Sandbox Code Playgroud)
这是Codepad上的代码.