0

0

JavaScript 对象属性分组:构建嵌套树形结构的完整教程

花韻仙語

花韻仙語

发布时间:2026-02-08 15:52:15

|

296人浏览过

|

来源于php中文网

原创

JavaScript 对象属性分组:构建嵌套树形结构的完整教程

本文介绍如何将扁平的 javascript 对象数组,按指定属性顺序(如 `['a', 'b', 'c', 'd']`)递归分组,生成符合图表库要求的嵌套树形结构(含 `name` 与 `children` 字段),支持任意层级、自动去重与深度优先展开。

数据可视化场景中(如 ECharts 的 tree 图、D3.js 层级布局或自定义钻取组件),常需将平面数据转换为具有明确父子关系的嵌套结构。本教程提供一种健壮、可扩展、无副作用的实现方案,核心思想分为两步:构建嵌套哈希树 → 递归展平为标准树节点

✅ 核心实现逻辑

我们不依赖外部库,仅用原生 JavaScript 实现:

  • 第一阶段(构建层级映射):遍历每个对象,按 keys 顺序(如 ['A','B','C','D'])逐层下钻,在内存中构建一个嵌套对象字典(类似 Trie 结构),键为对应属性值,值为下一层子对象;
  • 第二阶段(结构转换):通过递归函数 hierarchy() 将该字典转换为目标格式:若子对象非空,则生成带 children 的节点;若为空(即叶子节点),则只保留 name。

以下是完整可运行代码:

Sitekick
Sitekick

一个AI登陆页面自动构建器

下载
function groupByProperties(data, keys) {
  // 边界处理:空输入或无 keys
  if (!Array.isArray(data) || data.length === 0 || !Array.isArray(keys) || keys.length === 0) {
    return null;
  }

  // 构建嵌套字典:{ A1: { B5: { C7: { D67: {}, D69: {} } } } }
  const root = {};
  for (const item of data) {
    let node = root;
    for (const key of keys) {
      const value = item[key];
      // 确保 value 是合法 key(避免对象/undefined/null 作为 key)
      if (value == null || typeof value === 'object') {
        throw new Error(`Invalid grouping value at key "${key}": ${JSON.stringify(value)}`);
      }
      node = (node[value] ??= {});
    }
  }

  // 递归转换为 name/children 树结构
  const hierarchy = (obj) => 
    Object.entries(obj).map(([name, child]) => ({
      name: name,
      children: Object.keys(child).length > 0 ? hierarchy(child) : undefined
    })).map(node => 
      node.children === undefined ? { name: node.name } : node
    );

  const result = hierarchy(root);
  return result.length > 0 ? result[0] : { name: null };
}

// ✅ 示例 1:相同 A/B/C,不同 D
const data1 = [
  { A: 1, B: 5, C: 7, D: 67 },
  { A: 1, B: 5, C: 7, D: 69 }
];
console.log(groupByProperties(data1, ['A', 'B', 'C', 'D']));
// → { name: "1", children: [{ name: "5", children: [{ name: "7", children: [{ name: "67" }, { name: "69" }] }] }] }

// ✅ 示例 2:C 层级分化
const data2 = [
  { A: 1, B: 5, C: 4, D: 67 },
  { A: 1, B: 5, C: 7, D: 69 }
];
console.log(groupByProperties(data2, ['A', 'B', 'C', 'D']));
// → { name: "1", children: [{ name: "5", children: [{ name: "4", children: [{ name: "67" }] }, { name: "7", children: [{ name: "69" }] }] }] }

⚠️ 注意事项与最佳实践

  • 类型安全:代码中已校验 value 是否为有效对象键(字符串/数字/布尔值),禁止传入 null、undefined 或对象——否则会导致静默错误或意外分组;
  • 顺序敏感:keys 数组顺序决定树的层级深度与结构,务必严格按业务语义排列(如 ['region', 'category', 'year', 'quarter']);
  • 性能考量:时间复杂度为 O(n × m)(n=数据条数,m=keys长度),适用于万级以内数据;超大数据建议结合 Web Worker 或服务端预聚合;
  • 扩展性增强:如需支持自定义节点字段(如添加 id、label、metadata),可在 hierarchy 映射中注入额外属性;
  • 空值处理:当前策略抛出明确错误;生产环境可改为跳过非法项并记录警告(使用 console.warn)。

✅ 总结

该方案以“字典建模 + 递归展平”双阶段设计,兼顾可读性、健壮性与性能,完美适配各类需要层级化数据的前端图表库。只需传入原始数组和属性路径,即可获得开箱即用的树形结构,无需手动维护中间状态或嵌套循环,是处理多维分组需求的推荐范式。

立即学习Java免费学习笔记(深入)”;

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
ECharts是什么
ECharts是什么

ECharts是基于JavaScript的开源可视化库,能够帮助开发者轻松地实现各种复杂的数据可视化效果,提供了丰富的图表类型和交互功能。本专题为大家提供ECharts是什么的相关的文章、下载、课程内容,供大家免费下载体验。

273

2023.08.04

echarts自适应大小设置
echarts自适应大小设置

使用ECharts的自适应大小设置可以使图表能够根据不同屏幕尺寸和设备进行自适应。一种是使用resize事件,在图表容器大小改变时重新渲染图表;另一种是使用CSS样式,通过设置图表容器的宽度和高度为百分比值,使图表容器根据父元素的大小进行自适应。根据实际需求选择合适的方法,可以使图表在不同设备上都能够良好地显示和交互 。

470

2023.09.13

c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

241

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

641

2024.03.01

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

424

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

213

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1515

2023.10.24

字符串介绍
字符串介绍

字符串是一种数据类型,它可以是任何文本,包括字母、数字、符号等。字符串可以由不同的字符组成,例如空格、标点符号、数字等。在编程中,字符串通常用引号括起来,如单引号、双引号或反引号。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

634

2023.11.24

Golang处理数据库错误教程合集
Golang处理数据库错误教程合集

本专题整合了Golang数据库错误处理方法、技巧、管理策略相关内容,阅读专题下面的文章了解更多详细内容。

65

2026.02.06

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
React 教程
React 教程

共58课时 | 4.8万人学习

TypeScript 教程
TypeScript 教程

共19课时 | 2.8万人学习

Bootstrap 5教程
Bootstrap 5教程

共46课时 | 3.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号