【前端JS】List/Array数据 转 树形结构数据(适用于侧边栏菜单,多级分类等)

先上完整js代码供复制,再一步步分析思路,让你能明白怎么样思考才能对付这种递归类的算法处理

首先说明,函数命名很重要,因为某种程度上,命名的好坏能够决定你第一眼看到这个函数,能否想象或理解出它的大致实现思路。之前看网上一些写法,就是因为他们命名的问题,半天没理解这个思路,为什么可以这么做?他是怎么想到的。所以自己整理了此篇文章。并规范了函数命名
通常有两种叫法menu和submenus,parent和children,逻辑都一样,两种叫法都贴出来,复制后稍加修改即可

// 模拟数据
var menuList= [
  { 'id': '1', 'name': '动物', 'pid': '0' },
  { 'id': '2', 'name': '鸟类动物', 'pid': '1' },
  { 'id': '3', 'name': '鱼类动物', 'pid': '1' },
  { 'id': '4', 'name': '爬行类动物', 'pid': '1' },
  { 'id': '5', 'name': '杜鹃鸟', 'pid': '2' },
  { 'id': '6', 'name': '喜鹊', 'pid': '2' },
  { 'id': '7', 'name': '鲨鱼', 'pid': '3' },
  { 'id': '8', 'name': '比目鱼', 'pid': '3' },
  { 'id': '9', 'name': '鲸鱼', 'pid': '3' },
  { 'id': '10', 'name': '蜥蜴', 'pid': '4' }
]

/**
 * 封装完善子级菜单的方法
 * @param {obj} menu 父级菜单对象
 * @param {obj} menuList 菜单数据(list或array)
 */
function completeSubmenus(menu, menuList) {
  const submenus = []
  // 1.筛选出符合条件的子菜单
  for (const item of menuList) {
    if (item.pid === menu.id) {
      submenus.push(item)
    }
  }
  // 2.如果子菜单存在,则封装属性
  if (submenus.length > 0) {
    // 递归封装子菜单中的子子菜单
    for (const submenu of submenus) {
      completeSubmenus(submenu, menuList)
    }
    menu.submenus = submenus
  }
  // 3.返回封装后的menu对象
  return menu
}

// 测试代码
const topMenu = { 'id': '1', 'name': '动物', 'pid': '0' }
completeSubmenus(topMenu, menuList)
console.log(topMenu)
// 模拟数据
var moduleList = [
  { 'id': '1', 'name': '动物', 'pid': '0' },
  { 'id': '2', 'name': '鸟类动物', 'pid': '1' },
  { 'id': '3', 'name': '鱼类动物', 'pid': '1' },
  { 'id': '4', 'name': '爬行类动物', 'pid': '1' },
  { 'id': '5', 'name': '杜鹃鸟', 'pid': '2' },
  { 'id': '6', 'name': '喜鹊', 'pid': '2' },
  { 'id': '7', 'name': '鲨鱼', 'pid': '3' },
  { 'id': '8', 'name': '比目鱼', 'pid': '3' },
  { 'id': '9', 'name': '鲸鱼', 'pid': '3' },
  { 'id': '10', 'name': '蜥蜴', 'pid': '4' }
]

/**
 * 封装完善子级分类的方法
 * @param {obj} parentObj 父级分类对象
 * @param {obj} moduleList 分类模块数据(list或array)
 */
function completeChildren(parentObj, moduleList) {
  const children = []
  // 1.筛选出符合条件的子分类
  for (const item of moduleList) {
    if (item.pid === parentObj.id) {
      children.push(item)
    }
  }
  // 2.如果子分类存在,则封装属性
  if (children.length > 0) {
    // 递归封装子分类中的子子分类
    for (const item of children) {
      completeChildren(item, moduleList)
    }
    parentObj.children = children
  }
  // 3.返回封装后的parentObj对象
  return parentObj
}

// 测试代码
const parentObj = { 'id': '1', 'name': '动物', 'pid': '0' }
completeChildren(parentObj, moduleList)
console.log(parentObj)

分析

1.list转树形结构需求简介

list转树形结构是一种前端很常见的需求
比如角色管理的权限层级树,比如侧边导航栏的动态多级菜单等等,这种需求实在是太常见了

以角色管理的权限层级树为例,通常我们数据库设计这种有层级关系的数据结构,往往是有个子父级关系字段在里面
比如code与parent_code,id与pid等等,用来表示他们的父级归属关系
但是后端写的方法,往往返回给前端的就是这一堆的list对象数据,需要前端自己转化成树形结构


关系图.png
关系.png

2.原理分析

那么怎么转,怎么入手?
是个开发,都他娘的知道肯定要用上循环遍历或者递归调用之类的东西,但是说清楚这递归组装的思路,多少还是有些难度的
我们一步步入手:
先定义一个list数据

// 定义一个list数据
var menuList = [
  { 'id': '1', 'name': '动物', 'pid': '0' },
  { 'id': '2', 'name': '鸟类动物', 'pid': '1' },
  { 'id': '3', 'name': '鱼类动物', 'pid': '1' },
  { 'id': '4', 'name': '爬行类动物', 'pid': '1' },
  { 'id': '5', 'name': '杜鹃鸟', 'pid': '2' },
  { 'id': '6', 'name': '喜鹊', 'pid': '2' },
  { 'id': '7', 'name': '鲨鱼', 'pid': '3' },
  { 'id': '8', 'name': '比目鱼', 'pid': '3' },
  { 'id': '9', 'name': '鲸鱼', 'pid': '3' },
  { 'id': '10', 'name': '蜥蜴', 'pid': '4' }
]

思路第1步:如果要写个函数,封装顶级菜单层下的第一层子菜单,实现是不是很简单:把顶级菜单对象和list数据一起传入函数,遍历list,如果pid与顶级菜单id相等,就放入

function completeSubmenus(menu, menuList) {
  const submenus = []
  // 1.筛选出符合条件的子菜单
  for (const item of menuList) {
    if (item.pid === menu.id) {
      submenus.push(item)
    }
  }
  // 2.如果子菜单存在,则封装属性
  if (submenus.length > 0) {
    menu.submenus = submenus
  }
  // 3.返回封装后的menu对象
  return menu
}

const topMenu = { 'id': '1', 'name': '动物', 'pid': '0' }
completeSubmenus(topMenu, menuList)
console.log(topMenu)
测试结果1.png

思路第2步:一级子菜单既然能用该函数封装完成,那么遍历这些一级子菜单,调用相同的方法,是不是也能将这些子菜单的子子菜单一一封装完毕呢?答案是肯定的
我们稍微改下代码,加个递归即可

function completeSubmenus(menu, menuList) {
  const submenus = []
  // 1.筛选出符合条件的子菜单
  for (const item of menuList) {
    if (item.pid === menu.id) {
      submenus.push(item)
    }
  }
  // 2.如果子菜单存在,则封装属性
  if (submenus.length > 0) {
    // 递归封装子菜单中的子子菜单
    for (const submenu of submenus) {
      completeSubmenus(submenu, menuList)
    }
    menu.submenus = submenus
  }
  // 3.返回封装后的menu对象
  return menu
}

const topMenu = { 'id': '1', 'name': '动物', 'pid': '0' }
completeSubmenus(topMenu, menuList)
console.log(topMenu)
测试结果2.png

完毕

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,047评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,807评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,501评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,839评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,951评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,117评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,188评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,929评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,372评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,679评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,837评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,536评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,168评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,886评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,129评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,665评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,739评论 2 351