经典笔试题:扁平化数据结构转树

前言

最近在这三月份,陆陆续续接了了几个笔试题。其中有一个常考的题目。就是后台返回一个扁平的数据结构,转成树。

我们看下题目:打平的数据内容如下:

js 复制代码
let arr = [
    {id: 1, name: '部门1', pid: 0},
    {id: 2, name: '部门2', pid: 1},
    {id: 3, name: '部门3', pid: 1},
    {id: 4, name: '部门4', pid: 3},
    {id: 5, name: '部门5', pid: 4},
]

输出结果

js 复制代码
[
    {
        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: []
                            }
                        ]
                    }
                ]
            }
        ]
    }
]

解决思路

递归

主要思路是提供一个arrayToTree的方法,该方法递归去查找子集。

js 复制代码
/**
 * 递归查找
 */
function arrayToTree(array,pid){  
    let result = []  
    array.forEach(item=>{  
        if(item.pid==pid){  
            item.children = arrayToTree(array,item.id)  
            result.push(item)  
        }  
    })  
    return result  
}

从上面的代码我们分析,该实现的时间复杂度为O(2^n)你。

为什么时间复杂度是O(2^n)

首先,让我们来看函数中的主要操作:

  1. 遍历数组:代码中使用了 forEach 方法对数组进行了一次遍历。这一操作的时间复杂度为 O(n),其中 n 是数组的长度。
  2. 递归调用:在遍历数组的过程中,对于每个节点,都会进行递归调用 arrayToTree 函数来构建其子树。在最坏情况下,每次递归调用都需要遍历一次整个数组来寻找与当前节点的父节点 ID 相同的节点。

因此,递归调用的总时间复杂度为 O(n^2),其中 n 是数组的长度。

不用递归,如何优化

主要思路是先把数据转成Map去存储,之后遍历的同时借助对象的引用,直接从Map找对应的数据做存储。这样借助额外的空间,降低了时间复杂度

js 复制代码
function arrayToTree(items) {
  const result = []; // 存放结果
  const itemMap = {};

  // 先转成map存储
  for (const item of items) {
    itemMap[item.id] = { ...item, children: [] };
  }

  for (const item of items) {
    const id = item.id;
    const pid = item.pid;
    const treeItem = itemMap[id];

    if (pid === 0) {
      result.push(treeItem);
    } else {
      if (!itemMap[pid]) {
        itemMap[pid] = {
          children: [],
        };
      }
      itemMap[pid].children.push(treeItem);
    }
  }
  return result;
}

复杂度为多少

  1. 创建哈希表:在第一个循环中,我们遍历了一次输入的 items 数组,并将每个节点都添加到了哈希表 itemMap 中。这个操作的时间复杂度是 O(n),其中 n 是数组的长度。

  2. 构建树:在第二个循环中,我们再次遍历了一次 items 数组,对每个节点进行处理。对于每个节点,我们都需要访问其父节点,并将当前节点添加到父节点的 children 数组中。在访问和更新节点时,由于使用了哈希表,查找和访问节点的时间复杂度是 O(1),因此对每个节点的操作的时间复杂度也是 O(1)。因此,整个构建树的过程的时间复杂度是 O(n),其中 n 是数组的长度。

综上所述,整个算法的时间复杂度是 O(n),其中 n 是输入数组的长度。这是因为我们只需要遍历两次输入数组,并且在每次操作中都能以常数时间复杂度在哈希表中查找到对应的节点。。

结尾

如果你有着比以上更好的实现方法,欢迎在评论区留下你的答案,大家一起学习。

相关推荐
M_emory_5 分钟前
解决 git clone 出现:Failed to connect to 127.0.0.1 port 1080: Connection refused 错误
前端·vue.js·git
Ciito8 分钟前
vue项目使用eslint+prettier管理项目格式化
前端·javascript·vue.js
simple_ssn19 分钟前
【C语言刷力扣】1502.判断能否形成等差数列
c语言·算法·leetcode
寂静山林27 分钟前
UVa 11855 Buzzwords
算法
Curry_Math32 分钟前
LeetCode 热题100之技巧关卡
算法·leetcode
ahadee39 分钟前
蓝桥杯每日真题 - 第10天
c语言·vscode·算法·蓝桥杯
成都被卷死的程序员42 分钟前
响应式网页设计--html
前端·html
mon_star°1 小时前
将答题成绩排行榜数据通过前端生成excel的方式实现导出下载功能
前端·excel
Zrf21913184551 小时前
前端笔试中oj算法题的解法模版
前端·readline·oj算法
军训猫猫头1 小时前
35.矩阵格式的一到一百数字 C语言
c语言·算法