菜单应配置驱动、动态渲染,后端仅提供按角色查询的标准接口,返回含id、parentId、path等字段的树形数据;权限校验需前后端双重控制,路由须动态注册,确保配置、权限、路由三者语义严格对齐。

菜单数据必须与业务逻辑解耦,用配置驱动渲染
菜单不是硬编码在 Controller 或前端模板里的静态结构,而是由配置中心(如 Nacos、Apollo)或本地 application.yml / JSON 文件定义的树形数据。核心在于:后端只提供 MenuService.getMenusByRole(String role) 这类接口,返回标准 List,字段包含 id、parentId、path、component、permission 等——这些字段直接映射到前端路由/权限系统,不掺杂 Java 业务判断。
常见错误是把菜单生成逻辑写进 @Controller 方法里,比如用 if-else 拼 HTML 或手动 new Menu 对象。这会导致每次加菜单都要改代码、发版、重启服务。
- 菜单配置建议用 YAML 分层表达父子关系,避免手写 JSON 容易出错
-
path字段必须符合前端 Vue Router / React Router 的路径规则(如/user/list),不能带后缀或参数占位符 -
component值应为前端组件路径字符串(如"views/user/UserList.vue"),后端不做解析,只透传 - 权限字段
permission推荐用字符串数组(如["user:read", "user:export"]),方便前端 v-permission 指令消费
菜单实体需支持运行时动态组装,避免递归查询 N+1
数据库表设计通常为单表 sys_menu,含 id、parent_id、name、sort、is_hidden 等字段。查菜单时不能对每个节点都查一次子节点(即 for 循环里调 menuMapper.selectByParentId(id)),否则 5 层深的菜单会触发几十次 SQL。
正确做法是一次性查出全部有效菜单(按角色过滤 + 状态校验),再用 Java 内存组装树:
立即学习“Java免费学习笔记(深入)”;
public ListbuildMenuTree(List allMenus) { Map menuMap = allMenus.stream() .filter(m -> !m.getIsHidden()) .collect(Collectors.toMap(MenuDO::getId, this::convertToDTO)); List rootList = new ArrayList<>(); for (MenuDTO node : menuMap.values()) { Long parentId = node.getParentId(); if (parentId == null || parentId == 0 || !menuMap.containsKey(parentId)) { rootList.add(node); } else { menuMap.get(parentId).getChildren().add(node); } } return rootList; }
- 务必在 SQL 层就用
WHERE role_code IN (...)或关联sys_role_menu表完成权限过滤,不要查全量再 Java 过滤 -
sort字段用于同级排序,组装树后需对每个children列表调用Collections.sort(...),按sort升序 - DTO 中的
children字段类型必须是List,不能是List或泛型擦除后无法序列化的类型
权限校验不能只靠菜单配置,必须叠加接口级注解控制
菜单配置里的 permission 字段只决定“是否显示该菜单项”,不等于“用户能访问对应接口”。如果仅靠菜单配置做鉴权,攻击者绕过前端直接请求 /api/user/delete?id=1 就能越权。
必须在 Controller 方法上加权限注解,例如:
@DeleteMapping("/delete")
@PreAuthorize("hasPermission('user:delete')")
public Result delete(@RequestParam Long id) { ... }
-
@PreAuthorize需配合 Spring Security 自定义PermissionEvaluator实现,从当前登录用户中提取权限集合,比对字符串 - 菜单配置中的
permission和接口注解中的字符串必须完全一致(大小写、冒号、下划线敏感) - 前端点击菜单时,应校验当前用户是否拥有该菜单的
permission;但后端接口仍要二次校验,不可信任前端传来的任何标识 - 避免把权限字符串拼成动态表达式(如
"hasPermission('" + menu.getPermission() + "')"),易被注入
前端路由需由后端菜单数据自动生成,禁止手动维护 router/index.js
Vue 项目中,router/index.js 不应写死 routes: [ { path: '/user', component: UserList } ]。而应通过后端接口获取菜单后,用 router.addRoute() 动态注册:
// fetchMenus().then(menus => {
// menus.forEach(menu => {
// router.addRoute({
// path: menu.path,
// name: menu.name,
// component: () => import(`@/views${menu.component}`),
// meta: { permissions: menu.permission }
// })
// })
// })
这个过程的关键约束是:menu.component 必须是合法的相对路径,且前端工程中真实存在对应文件;否则 import() 会报 ChunkLoadError。
- 后端返回的
component字段不能带文件后缀(如不要"UserList.vue",而应是"user/UserList"),由前端统一补.vue - 路径别名(如
@/views)必须和前端 webpack/vite 配置一致,否则编译时报找不到模块 - 动态路由不会出现在
router.getRoutes()初始列表中,需要在守卫中检查是否已注册,未注册则先拉菜单再 addRoute,否则白屏
permission 字符串要在数据库字段、Java 注解、前端指令、打包路径四个地方同时生效且保持一致。任何一处命名偏差或空格遗漏,都会导致菜单不显示或接口 403。










