javascript将扁平的数据转为树形结构(高效率)

当我们需要将一个一维数组转换成一个多层结构的时候,最简单但是最慢的就是多个for循环嵌套,但是这样做有一些缺点,那就是效率太低、而且有多少层就需要嵌套几个for循环,不好用。

我实现了用O(n)级算法将 一个扁平的数组即一维数组代表的菜单结构转换成一个多层级的菜单结构。

一位数组中每一个元素必须要包含以下属性:

  1. 拥有一个唯一的id
  2. 拥有一个parent_id, 这个id指向它父级的id

其他则为每一个元素中的一些信息,我这里是菜单,就有菜单的名称和url信息。

注:

  1. 在层级结构中,第一层的parent_id需要为0.
  2. 父节点在数组中的位置需要在子节点前,即 节点3必须排在节点3-2之前

扁平数组例:

var menu_list = [{
      id: '1',
      menu_name: '设置',
      menu_url: 'setting',
      parent_id: 0
    }, {
      id: '1-1',
      menu_name: '权限设置',
      menu_url: 'setting.permission',
      parent_id: '1'
    }, {
      id: '1-1-1',
      menu_name: '用户管理列表',
      menu_url: 'setting.permission.user_list',
      parent_id: '1-1'
    }, {
      id: '1-1-2',
      menu_name: '用户管理新增',
      menu_url: 'setting.permission.user_add',
      parent_id: '1-1'
    }, {
      id: '1-1-3',
      menu_name: '角色管理列表',
      menu_url: 'setting.permission.role_list',
      parent_id: '1-1'
    }, {
      id: '1-2',
      menu_name: '菜单设置',
      menu_url: 'setting.menu',
      parent_id: '1'
    }, {
      id: '1-2-1',
      menu_name: '菜单列表',
      menu_url: 'setting.menu.menu_list',
      parent_id: '1-2'
    }, {
      id: '1-2-2',
      menu_name: '菜单添加',
      menu_url: 'setting.menu.menu_add',
      parent_id: '1-2'
    }, {
      id: '2',
      menu_name: '订单',
      menu_url: 'order',
      parent_id: 0
    }, {
      id: '2-1',
      menu_name: '报单审核',
      menu_url: 'order.orderreview',
      parent_id: '2'
    }, {
      id: '2-2',
      menu_name: '退款管理',
      menu_url: 'order.refundmanagement',
      parent_id: '2'
    }
]

实现算法buildTree

算法思想:

  1. 先将数组中的每一个节点放到temp对象中(创建set)
    即数组中有{id: '2-3', parent_id: '2',...}这样一个节点,需要将他放到temp中变成 '2-3': {id: '2-3', parent_id: '2',...}这种JSON结构
  2. 直接遍历整个temp对象,通过这句代码   temp[temp[i].parent_id].children[temp[i].id] = temp[i];   将当前子节点与父节点建立连接。是因为我们保证了父节点一定在子节点前,那么当子节点出现的时候就直接可以用temp[temp[i].parent_id]来查找到父节点这个时候先父节点的children对象中添加一个引用即可。
/**
 * 将一维的扁平数组转换为多层级对象
 * @param  {[type]} list 一维数组,数组中每一个元素需包含id和parent_id两个属性 
 * @return {[type]} tree 多层级树状结构
 */
function buildTree(list){
	let temp = {};
	let tree = {};
	for(let i in list){
		temp[list[i].id] = list[i];
	}
	for(let i in temp){
		if(temp[i].parent_id) {
			if(!temp[temp[i].parent_id].children) {
				temp[temp[i].parent_id].children = new Object();
			}
			temp[temp[i].parent_id].children[temp[i].id] = temp[i];
		} else {
			tree[temp[i].id] =  temp[i];
		}
	}
	return tree;
}

测试结果:

可以看到函数成功地构建了多级的树状结构

 

这个算法的效率是极高的,比多重for循环来的好得多。

 

 

 

 

以下是测试数据,用时只需5毫秒左右:


var menu_list = [{
      id: '1',
      menu_name: '设置',
      menu_url: 'setting',
      parent_id: 0
    }, {
      id: '1-1',
      menu_name: '权限设置',
      menu_url: 'setting.permission',
      parent_id: '1'
    }, {
      id: '1-1-1',
      menu_name: '用户管理列表',
      menu_url: 'setting.permission.user_list',
      parent_id: '1-1'
    }, {
      id: '1-1-2',
      menu_name: '用户管理新增',
      menu_url: 'setting.permission.user_add',
      parent_id: '1-1'
    }, {
      id: '1-1-3',
      menu_name: '角色管理列表',
      menu_url: 'setting.permission.role_list',
      parent_id: '1-1'
    }, {
      id: '1-1-4',
      menu_name: '角色管理新增',
      menu_url: 'setting.permission.role_add',
      parent_id: '1-1'
    }, {
      id: '1-2',
      menu_name: '菜单设置',
      menu_url: 'setting.menu',
      parent_id: '1'
    }, {
      id: '1-2-1',
      menu_name: '菜单列表',
      menu_url: 'setting.menu.menu_list',
      parent_id: '1-2'
    }, {
      id: '1-2-2',
      menu_name: '菜单添加',
      menu_url: 'setting.menu.menu_add',
      parent_id: '1-2'
    }, {
      id: '2',
      menu_name: '订单',
      menu_url: 'order',
      parent_id: 0
    }, {
      id: '2-1',
      menu_name: '报单审核',
      menu_url: 'order.orderreview',
      parent_id: '2'
    }, {
      id: '2-2',
      menu_name: '退款管理',
      menu_url: 'order.refundmanagement',
      parent_id: '2'
    }, {
      id: '2-3',
      menu_name: '实物订单',
      menu_url: 'order.realorder',
      parent_id: '2'
    }, {
      id: '2-1-1',
      menu_name: '全部报单',
      menu_url: 'order.orderreview.all',
      parent_id: '2-1'
    }, {
      id: '2-2-1',
      menu_name: '所有记录',
      menu_url: 'order.refundmanagement.all',
      parent_id: '2-2'
    }, {
      id: '2-2-2',
      menu_name: '待处理',
      menu_url: 'order.refundmanagement.wait',
      parent_id: '2-2'
    }, {
      id: '2-2-3',
      menu_name: '退款原因',
      menu_url: 'order.refundmanagement.result',
      parent_id: '2-2'
    }, {
      id: '2-3-1',
      menu_name: '实物订单管理',
      menu_url: 'order.realorder.list',
      parent_id: '2-3'
    }, {
      id: '3',
      menu_name: '商品',
      menu_url: 'commodity',
      parent_id: 0
    }, {
      id: '3-1',
      menu_name: '分类管理',
      menu_url: 'commodity.classifieldmanagement',
      parent_id: '3'
    }, {
      id: '3-1-1',
      menu_name: '管理',
      menu_url: 'commodity.classifieldmanagement.management',
      parent_id: '3-1'
    }, {
      id: '3-1-2',
      menu_name: '编辑或新增',
      menu_url: 'commodity.classifieldmanagement.edit',
      parent_id: '3-1'
    }, {
      id: '3-2',
      menu_name: '品牌管理',
      menu_url: 'commodity.brandmanagement',
      parent_id: '3'
    }, {
      id: '3-2-1',
      menu_name: '管理',
      menu_url: 'commodity.brandmanagement.management',
      parent_id: '3-2'
    }, {
      id: '3-2-2',
      menu_name: '编辑或新增',
      menu_url: 'commodity.brandmanagement.edit',
      parent_id: '3-2'
    }, {
      id: '3-3',
      menu_name: '商品管理',
      menu_url: 'commodity.commoditymanagement',
      parent_id: '3'
    }, {
      id: '3-3-1',
      menu_name: '管理',
      menu_url: 'commodity.commoditymanagement.management',
      parent_id: '3-3'
    }, {
      id: '3-3-2',
      menu_name: '编辑或新增',
      menu_url: 'commodity.commoditymanagement.edit',
      parent_id: '3-3'
    }, {
      id: '3-4',
      menu_name: '类型管理',
      menu_url: 'commodity.typeManagement',
      parent_id: '3'
    }, {
      id: '3-4-1',
      menu_name: '管理',
      menu_url: 'commodity.typeManagement.management',
      parent_id: '3-4'
    }, {
      id: '3-4-2',
      menu_name: '编辑或新增',
      menu_url: 'commodity.typeManagement.edit',
      parent_id: '3-4'
    }];

这是我一个大二学生想出来的,挺开心的,因为当时看到老师用的3个for循环嵌套。嘿嘿嘿 

  • 18
    点赞
  • 38
    收藏
    觉得还不错? 一键收藏
  • 12
    评论
扁平数据指的是将具有层级关系的数据展开成一维结构,而树形结构则是将具有层级关系的数据呈现为状图。因此,将扁平数据转换成树形结构需要进行一些处理。 以下是一个示例数据: ```json [ { "id": 1, "name": "节点1", "parentId": null }, { "id": 2, "name": "节点2", "parentId": 1 }, { "id": 3, "name": "节点3", "parentId": 2 }, { "id": 4, "name": "节点4", "parentId": 2 }, { "id": 5, "name": "节点5", "parentId": 1 }, { "id": 6, "name": "节点6", "parentId": 5 } ] ``` 其中,每个对象表示一个节点,`id`表示节点的唯一标识,`name`表示节点的名称,`parentId`表示节点的父节点标识。其中,根节点的`parentId`为`null`。 现在要将这个扁平数据转换成树形结构,可以按照以下步骤进行: 1. 创建一个空的树形结构对象 2. 遍历扁平数据,将每个节点加入树形结构中 3. 对于每个节点,如果它是根节点,则直接加入树形结构对象的`children`属性中,否则需要将它加入它的父节点的`children`属性中 以下是使用JavaScript实现的代码: ```javascript function flatToTree(flatData) { const tree = {}; const map = {}; for (const node of flatData) { const id = node.id; const parentId = node.parentId; if (!map[id]) { map[id] = { children: [] }; } map[id].id = id; map[id].name = node.name; if (parentId === null) { tree[id] = map[id]; } else { if (!map[parentId]) { map[parentId] = { children: [] }; } map[parentId].children.push(map[id]); } } return Object.values(tree); } ``` 使用该函数对示例数据进行转换,可以得到如下结果: ```json [ { "id": 1, "name": "节点1", "children": [ { "id": 2, "name": "节点2", "children": [ { "id": 3, "name": "节点3", "children": [] }, { "id": 4, "name": "节点4", "children": [] } ] }, { "id": 5, "name": "节点5", "children": [ { "id": 6, "name": "节点6", "children": [] } ] } ] } ] ``` 可以看到,该函数将扁平数据成功转换成了树形结构

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 12
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值