使用递归函数可将数据库中的父子结构菜单转化为HTML嵌套列表。1. 创建包含id、name、url、parent_id、sort_order字段的menus表,parent_id指向父级,顶级为0;2. 插入示例数据构建层级关系;3. 通过buildMenuTree函数递归组织数据为树形结构;4. 利用renderMenu函数递归生成ul/li HTML输出;5. 可扩展样式、权限控制、缓存等优化。该方法适用于无限级导航菜单的动态渲染。

在构建动态菜单时,尤其是多级嵌套的导航菜单(如后台管理系统、网站栏目等),数据往往以父子关系存储。使用 PHP 递归函数可以轻松将这种结构转化为 HTML 嵌套列表。下面介绍如何基于数据库结构和递归逻辑实现一个可扩展的动态菜单系统。
准备数据库结构
为了支持无限层级的菜单,数据库表需要包含自引用字段。例如创建一个 menus 表:
CREATE TABLE menus (
id INT PRIMARY KEY AUTO_INCREMENT,
name VARCHAR(100) NOT NULL,
url VARCHAR(255),
parent_id INT DEFAULT 0,
sort_order INT DEFAULT 0
);
其中 parent_id 指向父级菜单的 id,顶级菜单的 parent_id 为 0。
示例数据:
立即学习“PHP免费学习笔记(深入)”;
INSERT INTO menus (id, name, url, parent_id, sort_order) VALUES (1, '首页', '/', 0, 1), (2, '产品中心', '/products', 0, 2), (3, '家用电器', '/products/appliances', 2, 1), (4, '手机数码', '/products/digital', 2, 2), (5, '关于我们', '/about', 0, 3), (6, '公司简介', '/about#intro', 5, 1), (7, '联系我们', '/about#contact', 5, 2);
获取菜单数据并组织成树形结构
先从数据库中读取所有菜单项,然后通过递归函数将其构造成嵌套数组。
function buildMenuTree($data, $parentId = 0) {
$tree = [];
foreach ($data as $item) {
if ($item['parent_id'] == $parentId) {
$children = buildMenuTree($data, $item['id']);
if (!empty($children)) {
$item['children'] = $children;
}
$tree[] = $item;
}
}
return $tree;
}
调用方式:
// 假设已连接数据库
$stmt = $pdo->query("SELECT * FROM menus ORDER BY parent_id, sort_order");
$menuItems = $stmt->fetchAll(PDO::FETCH_ASSOC);
$menuTree = buildMenuTree($menuItems);
递归输出HTML菜单
有了树形结构后,使用另一个递归函数生成 HTML 的 ul/li 结构。
function renderMenu($menuTree) {
if (empty($menuTree)) return '';
$html = '
- ';
foreach ($menuTree as $item) {
$html .= '
- ' . htmlspecialchars($item['name']) . ''; if (isset($item['children']) && is_array($item['children'])) { $html .= renderMenu($item['children']); // 递归调用 } $html .= ' '; } $html .= '
}
// 输出菜单 echo renderMenu($menuTree);
优化与扩展建议
实际项目中可根据需求进行以下改进:
- 添加 CSS 类名控制样式,如当前页面高亮
- 增加权限判断,在递归中过滤无访问权限的菜单项
- 缓存菜单树结构,避免每次请求都查询数据库
- 支持图标、是否新窗口打开等字段扩展
基本上就这些。通过递归函数处理父子结构数据,是实现动态菜单最清晰有效的方式之一,尤其适合栏目、分类、权限菜单等场景。只要理解“找子元素 → 递归渲染”的逻辑,就能灵活应用到各种嵌套结构中。











