gpt4 book ai didi

PHP - 从平面数组制作嵌套树菜单结构

转载 作者:可可西里 更新时间:2023-11-01 13:27:26 24 4
gpt4 key购买 nike

我正在根据从 WP 数据库获得的响应制作一个嵌套菜单数组。在 corcel package 的帮助下,我正在 Laravel 的 Controller 中从 WP 获取数据。 , 然后用菜单数据制作一个数组,现在是一层深。因此,当菜单链接有子菜单链接时,数组如下所示:

{
"Hjem": {
"ID": 112,
"title": "Hjem",
"slug": "hjem",
"url": "http://hivnorge.app/?p=112",
"status": "publish",
"main_category": "Hovedmeny",
"submenus": [
{
"ID": 129,
"title": "Lorem ipsum",
"slug": "lorem-ipsum",
"url": "http://hivnorge.app/?p=129",
"status": "publish",
"main_category": "Nyheter"
}
]
},
"Nytt test innlegg": {
"ID": 127,
"title": "Nytt test innlegg",
"slug": "nytt-test-innlegg",
"url": "http://hivnorge.app/?p=127",
"status": "private",
"main_category": "Nyheter",
"submenus": [
{
"ID": 125,
"title": "Test innlegg",
"slug": "test-innlegg",
"url": "http://hivnorge.app/?p=125",
"status": "publish",
"main_category": "Nyheter"
},
{
"ID": 129,
"title": "Lorem ipsum",
"slug": "lorem-ipsum",
"url": "http://hivnorge.app/?p=129",
"status": "publish",
"main_category": "Nyheter"
}
]
},
"Prosjektsamarbeidets verdi": {
"ID": 106,
"title": "Prosjektsamarbeidets verdi",
"slug": "prosjektsamarbeidets-verdi",
"url": "http://hivnorge.no.wordpress.seven.fredrikst/?p=106",
"status": "publish",
"main_category": "Prevensjon"
}
}

这就是我创建此响应的方式:

        $menu = Menu::slug('hovedmeny')->first();
$res = [];

foreach ($menu->nav_items as $item) {
$item->makeHidden($hiddenAttributes)->toArray();
$parent_id = $item->meta->_menu_item_menu_item_parent;

if ($parent_id == '0') {
if ($item->title == '') {
$item = $this->findPost($item);
}
$parentItem = $item;
$res[$parentItem->title] = $parentItem->makeHidden($hiddenAttributes)->toArray();
}
else {
$childItem = $this->findPost($item);
$res[$parentItem->title]['submenus'][] = $childItem->makeHidden($hiddenAttributes)->toArray();
}
}

return $res;

我遇到的问题是 WP 的响应只为每个 $item 返回 parent_id 并且没有关于一个项目是否有一些 child 的数据,所以这是元数据父项的数据例如:

         #attributes: array:4 [
"meta_id" => 209
"post_id" => 112
"meta_key" => "_menu_item_menu_item_parent"
"meta_value" => "0"
]

这是子项的元数据:

          #attributes: array:4 [
"meta_id" => 326
"post_id" => 135
"meta_key" => "_menu_item_menu_item_parent"
"meta_value" => "112"
]

我怎样才能使它更灵活并启用更深的嵌套,以便我可以在子菜单中包含子菜单?

我试图寻找解决方案 here ,因为这与我的问题几乎相同,但无法实现。在我的数组菜单项中也只有 parent_id0parent_id 被视为根元素。还有 parent_id 如果 menu item 是一个 post,指向 meta id,而不是 id我需要的 post,所以我需要从 meta->_menu_item_object_id 中获取它。

更新

我已经设法制作了一个树状结构,但我现在遇到的问题是我不知道如何为 posts title。我在前面的示例中通过检查 title 是否为空来做到这一点,然后我将通过 id 搜索该 post:

          if ($item->title == '') {
$item = $this->findPost($item);
}

但是,对于新代码,我正在制作树状结构,但我不确定该怎么做,从那时起我无法制作树结构,因为我正在将所有内容与 id 进行比较,并且菜单元素的 ids 与指向的 postid 不同,所以我是无法制作树结构:

    private function menuBuilder($menuItems, $parentId = 0)
{
$hiddenAttributes = \Config::get('middleton.wp.menuHiddenAttributes');
$res = [];

foreach ($menuItems as $index => $item) {
$itemParentId = $item->meta->_menu_item_menu_item_parent;

if ($itemParentId == $parentId) {
$children = self::menuBuilder($menuItems, $item->ID);

if ($children) {
$item['submenu'] = $children;
}

$res[$item->ID] = $item->makeHidden($hiddenAttributes)->toArray();
unset($menuItems[$index]);
}
}

return $res;
}

那么,那么我得到的数据是:

   {
"112": {
"ID": 112,
"submenu": {
"135": {
"ID": 135,
"title": "",
"slug": "135",
"url": "http://hivnorge.app/?p=135",
"status": "publish",
"main_category": "Hovedmeny"
}
},
"title": "Hjem",
"slug": "hjem",
"url": "http://hivnorge.app/?p=112",
"status": "publish",
"main_category": "Hovedmeny"
},
"136": {
"ID": 136,
"submenu": {
"137": {
"ID": 137,
"submenu": {
"138": {
"ID": 138,
"title": "",
"slug": "138",
"url": "http://hivnorge.app/?p=138",
"status": "publish",
"main_category": "Hovedmeny"
}
},
"title": "",
"slug": "137",
"url": "http://hivnorge.app/?p=137",
"status": "publish",
"main_category": "Hovedmeny"
}
},
"title": "",
"slug": "136",
"url": "http://hivnorge.app/?p=136",
"status": "publish",
"main_category": "Hovedmeny"
},
"139": {
"ID": 139,
"title": "",
"slug": "139",
"url": "http://hivnorge.app/?p=139",
"status": "publish",
"main_category": "Hovedmeny"
}
}

最佳答案

解决这个问题的一种方法是使用变量别名。如果您注意管理 ID 的查找表(数组),您可以使用它插入到分层菜单数组的正确位置,因为不同的变量(这里是查找表中的数组条目)可以引用相同的值.

在下面的示例中对此进行了演示。它还解决了第二个问题(隐含在您的问题中),即平面数组未排序(顺序在数据库结果表中未定义),因此子菜单条目可以在菜单之前的结果集中子菜单条目所属的条目。

例如,我创建了一个简单的平面数组:

# some example rows as the flat array
$rows = [
['id' => 3, 'parent_id' => 2, 'name' => 'Subcategory A'],
['id' => 1, 'parent_id' => null, 'name' => 'Home'],
['id' => 2, 'parent_id' => null, 'name' => 'Categories'],
['id' => 4, 'parent_id' => 2, 'name' => 'Subcategory B'],
];

然后要完成的工作有两个主要变量:第一个是要创建的分层数组 $menu,第二个是查找表的 $byId:

# initialize the menu structure
$menu = []; # the menu structure
$byId = []; # menu ID-table (temporary)

lookup table 只在菜单建好时才需要,建好之后就扔掉了。

下一个重要步骤是通过遍历平面数组来创建 $menu。这是一个更大的 foreach 循环:

# build the menu (hierarchy) from flat $rows traversable
foreach ($rows as $row) {
# map row to local ID variables
$id = $row['id'];
$parentId = $row['parent_id'];

# build the entry
$entry = $row;
# init submenus for the entry
$entry['submenus'] = &$byId[$id]['submenus']; # [1]

# register the entry in the menu structure
if (null === $parentId) {
# special case that an entry has no parent
$menu[] = &$entry;
} else {
# second special case that an entry has a parent
$byId[$parentId]['submenus'][] = &$entry;
}

# register the entry as well in the menu ID-table
$byId[$id] = &$entry;

# unset foreach (loop) entry alias
unset($entry);
}

这是将条目从平面数组 ( $rows ) 映射到分层 $menu 数组的地方。由于堆栈和查找表 $byId,不需要递归。

这里的关键点是在将新条目添加到 $menu 结构时以及将它们添加到 $byId 时使用变量别名(引用)。这允许使用两个不同的变量名访问内存中的相同值:

        # special case that an entry has no parent
$menu[] = &$entry;
...

# register the entry as well in the menu ID-table
$byId[$id] = &$entry;

这是通过 = & 分配完成的,这意味着 $byId[$id] 可以访问 $menu[<< new key >>]

如果将其添加到子菜单,则执行相同的操作:

    # second special case that an entry has a parent
$byId[$parentId]['submenus'][] = &$entry;
...

# register the entry as well in the menu ID-table
$byId[$id] = &$entry;

这里 $byId[$id] 指向 $menu...[ << parent id entry in the array >>]['submenus'][ << new key >> ]

这解决了始终找到将新条目插入层次结构的正确位置的问题。

为了处理子菜单在其所属的菜单条目之前出现在平面数组中的情况,需要在为新条目初始化时从查找表中取出子菜单(在 [ 1]):

# init submenus for the entry
$entry['submenus'] = &$byId[$id]['submenus']; # [1]

这有点特殊。如果 $byId[$id]['submenus'] 尚未设置(例如在第一个循环中),由于引用(null 前面的 & ),它被隐式设置为 &$byId[$id]['submenus'] 。如果已设置,来自尚未存在的条目的现有子菜单将用于初始化条目的子菜单。

这样做就足以不依赖于 $rows 中的任何特定顺序。

这就是循环的作用。

剩下的就是清理工作了:

# unset ID aliases
unset($byId);

它取消设置外观 ID 表,因为不再需要它。也就是说,所有别名都未设置。

完成示例:

# visualize the menu structure
print_r($menu);

然后给出以下输出:

Array
(
[0] => Array
(
[id] => 1
[parent_id] =>
[name] => Home
[submenus] =>
)

[1] => Array
(
[id] => 2
[parent_id] =>
[name] => Categories
[submenus] => Array
(
[0] => Array
(
[id] => 3
[parent_id] => 2
[name] => Subcategory A
[submenus] =>
)

[1] => Array
(
[id] => 4
[parent_id] => 2
[name] => Subcategory B
[submenus] =>
)

)

)

)

我希望这是可以理解的,并且您能够将其应用到您的具体场景中。您可以将它包装在它自己的函数中(我会建议这样做),我只是为了更好地演示这些部分而在示例中保持冗长。

相关问答资料:

关于PHP - 从平面数组制作嵌套树菜单结构,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/44779734/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com