js中处理树形数据

平时开发中会经常碰到树形数据,而我们存储的数据往往是扁平的数据,需要在前端或者后端对数据进行进一步处理成前端组件需要的树形数据。在操作完树形数据后再转换成扁平数据发给后端。下面JavaScript 中处理树形数据的方法,包括构建树和解析树。

一、构建树

假设我们有一个扁平的数组数据,每个元素包含 idparentIdname 等属性,我们可以将其构建为树形结构。以下是一个实现函数:

javascript 复制代码
function buildTree(arr, parentId = null) {
  let tree = [];
  arr.forEach(item => {
    if (item.parentId === parentId) {
      let children = buildTree(arr, item.id);
      if (children.length > 0) {
        item.children = children;
      }
      tree.push(item);
    }
  });
  return tree;
}

以下是使用示例:

javascript 复制代码
let flatData = [
  { id: 1, parentId: null, name: 'Root' },
  { id: 2, parentId: 1, name: 'Child 1' },
  { id: 3, parentId: 1, name: 'Child 2' },
  { id: 4, parentId: 3, name: 'Grandchild 1' }
];
let treeData = buildTree(flatData);
console.log(treeData);

解释:

  • buildTree 函数接收一个扁平数组 arr 和一个可选的 parentId 作为参数,默认为 null,用于表示根节点。
  • 函数使用 forEach 遍历数组中的每个元素。
  • 如果元素的 parentId 等于传入的 parentId,则递归调用 buildTree 函数查找其子节点,并将结果存储在 children 属性中。
  • 最终将满足条件的元素添加到 tree 数组中。

二、解析树

假设我们有一个树形结构的数据,我们要将其解析为一个扁平的数组。以下是一个实现函数:

javascript 复制代码
function flattenTree(tree, parentId = null) {
  let flatArray = [];
  tree.forEach(item => {
    let newItem = {...item };
    newItem.parentId = parentId;
    flatArray.push(newItem);
    if (item.children) {
      flatArray = flatArray.concat(flattenTree(item.children, item.id));
    }
  });
  return flatArray;
}

以下是使用示例:

javascript 复制代码
let treeData = [
  {
    id: 1,
    name: 'Root',
    children: [
      { id: 2, name: 'Child 1' },
      {
        id: 3,
        name: 'Child 2',
        children: [
          { id: 4, name: 'Grandchild 1' }
        ]
      }
    ]
  }
];
let flatData = flattenTree(treeData);
console.log(flatData);

解释:

  • flattenTree 函数接收一个树形结构 tree 和一个可选的 parentId 作为参数,默认为 null
  • 函数使用 forEach 遍历树中的每个元素。
  • 首先将当前元素复制到 newItem 中,并设置 parentId
  • newItem 加入到 flatArray 中。
  • 如果元素有 children,递归调用 flattenTree 函数将子节点添加到 flatArray 中,并将当前元素的 id 作为子节点的 parentId
相关推荐
微小冷2 小时前
Rust异步编程详解
开发语言·rust·async·await·异步编程·tokio
CappuccinoRose2 小时前
JavaScript 学习文档(二)
前端·javascript·学习·数据类型·运算符·箭头函数·变量声明
A9better2 小时前
C++——不一样的I/O工具与名称空间
开发语言·c++·学习
清水白石0082 小时前
《为什么说 deque 是 Python 滑动窗口的“隐藏神器”?深入解析双端队列的高效之道》
开发语言·python
杜子不疼.2 小时前
Ascend_C自定义算子开发
c语言·开发语言
WooaiJava2 小时前
流式TTS音频播放项目 - 面试问答(后端)
java·开发语言
全栈前端老曹2 小时前
【MongoDB】深入研究副本集与高可用性——Replica Set 架构、故障转移、读写分离
前端·javascript·数据库·mongodb·架构·nosql·副本集
新缸中之脑2 小时前
开发AI代理必备的8个Python 库
开发语言·人工智能·python
暴走十八步2 小时前
PHP+vscode开启调试debug
开发语言·vscode·php
郝学胜-神的一滴2 小时前
Python 列表 vs 数组:深入解析与最佳选择指南
开发语言·python·程序人生