0

0

使用Underscore.js处理嵌套数组数据并统计元素出现频率

聖光之護

聖光之護

发布时间:2025-09-30 09:15:33

|

161人浏览过

|

来源于php中文网

原创

使用Underscore.js处理嵌套数组数据并统计元素出现频率

本文详细介绍了如何利用Underscore.js高效地从嵌套数组中提取数据并统计元素的出现频率。通过结合_.map()、_.flatten()和_.countBy()等方法,可以简洁地实现这一目标。文章还探讨了JavaScript原生flatMap()的用法,并深入分析了_.reduce()在实现此功能时的常见误区及正确实践,旨在帮助开发者选择最合适的工具和方法来处理复杂数据结构。

场景描述

在数据处理中,我们经常会遇到需要从复杂数据结构中提取特定信息并进行统计的场景。例如,给定一个包含多个对象的数组,每个对象又包含一个子数组,我们需要统计子数组中元素的出现频率。以下是一个具体的例子:我们有一个nfl球队的数组,每个球队对象中包含一个球员名字的首字母数组。我们的目标是统计所有球员名字首字母的出现次数,并将其存储在一个对象中,例如:{'joe': 2, 'jimmy': 1, 'jalen': 1, ...}。

初始数据结构如下:

var nflTeams = [
  { name: 'Kansas City Chiefs', playersFirstNames: ['Shane', 'Chad', 'Michael', 'Ronald', 'Blake', 'Noah'], champions: true },
  { name: 'Philadelphia Eagles', playersFirstNames: ['Jalen', 'Kenneth', 'Boston', 'Trey', 'Jack', 'Andre', 'Jack', 'Lane', 'Jason', 'Nakobe'], champions: false },
  { name: 'Cincinnati Bengals', playersFirstNames: ['Brandon', 'Joe', 'Chris', 'Joe', 'Tyler', 'Trenton', 'Trent', 'Mitchell', 'Alex', 'Trey', 'Ted'], champions: false },
  { name: 'San Francisco 49ers', playersFirstNames: ['Jimmy', 'Josh', 'Kyle', 'Jordan', 'Brandon', 'Danny', 'George', 'Tyler', 'Charlie', 'Jake', 'Nick', 'Nick', 'Kevin'], champions: false },
];

推荐方案:利用Underscore.js的_.countBy()

Underscore.js提供了_.countBy()方法,专门用于统计集合中每个元素出现的次数。结合数据扁平化操作,可以非常简洁地实现我们的目标。

1. 使用JavaScript原生flatMap()与_.countBy()

如果您的运行环境支持ES2019及更高版本,可以使用数组原生的flatMap()方法。flatMap()可以先对数组中的每个元素执行map操作,然后将结果扁平化为一级新数组。这在处理嵌套数组时非常方便。

const nflTeams = [
  { name: 'Kansas City Chiefs', playersFirstNames: ['Shane', 'Chad', 'Michael', 'Ronald', 'Blake', 'Noah'], champions: true },
  { name: 'Philadelphia Eagles', playersFirstNames: ['Jalen', 'Kenneth', 'Boston', 'Trey', 'Jack', 'Andre', 'Jack', 'Lane', 'Jason', 'Nakobe'], champions: false },
  { name: 'Cincinnati Bengals', playersFirstNames: ['Brandon', 'Joe', 'Chris', 'Joe', 'Tyler', 'Trenton', 'Trent', 'Mitchell', 'Alex', 'Trey', 'Ted'], champions: false },
  { name: 'San Francisco 49ers', playersFirstNames: ['Jimmy', 'Josh', 'Kyle', 'Jordan', 'Brandon', 'Danny', 'George', 'Tyler', 'Charlie', 'Jake', 'Nick', 'Nick', 'Kevin'], champions: false },
];

// 导入 Underscore.js 库
// 

const resWithFlatMap = _.countBy(nflTeams.flatMap(team => team.playersFirstNames));
console.log(resWithFlatMap);
// 预期输出: { 'Shane': 1, 'Chad': 1, 'Michael': 1, 'Ronald': 1, 'Blake': 1, 'Noah': 1, 'Jalen': 1, 'Kenneth': 1, 'Boston': 1, 'Trey': 2, 'Jack': 2, 'Andre': 1, 'Lane': 1, 'Jason': 1, 'Nakobe': 1, 'Brandon': 1, 'Joe': 2, 'Chris': 1, 'Tyler': 2, 'Trenton': 1, 'Trent': 1, 'Mitchell': 1, 'Alex': 1, 'Ted': 1, 'Jimmy': 1, 'Josh': 1, 'Kyle': 1, 'Jordan': 1, 'Danny': 1, 'George': 1, 'Charlie': 1, 'Jake': 1, 'Nick': 2, 'Kevin': 1 }

注意事项: Underscore.js本身不提供flatMap方法,但Lodash等其他工具库或现代JavaScript环境提供了此功能。

2. 链式调用_.map()、_.flatten()和_.countBy()

如果需要纯粹使用Underscore.js的方法,或者环境不支持flatMap(),可以通过_.chain()、_.map()和_.flatten()的组合来实现相同的数据扁平化效果,然后再使用_.countBy()进行统计。

const nflTeams = [
  { name: 'Kansas City Chiefs', playersFirstNames: ['Shane', 'Chad', 'Michael', 'Ronald', 'Blake', 'Noah'], champions: true },
  { name: 'Philadelphia Eagles', playersFirstNames: ['Jalen', 'Kenneth', 'Boston', 'Trey', 'Jack', 'Andre', 'Jack', 'Lane', 'Jason', 'Nakobe'], champions: false },
  { name: 'Cincinnati Bengals', playersFirstNames: ['Brandon', 'Joe', 'Chris', 'Joe', 'Tyler', 'Trenton', 'Trent', 'Mitchell', 'Alex', 'Trey', 'Ted'], champions: false },
  { name: 'San Francisco 49ers', playersFirstNames: ['Jimmy', 'Josh', 'Kyle', 'Jordan', 'Brandon', 'Danny', 'George', 'Tyler', 'Charlie', 'Jake', 'Nick', 'Nick', 'Kevin'], champions: false },
];

// 导入 Underscore.js 库
// 

const resWithChain = _.chain(nflTeams)
  .map('playersFirstNames') // 提取所有球队的 playersFirstNames 数组
  .flatten()               // 将所有子数组扁平化为一个单一的数组
  .countBy()               // 统计每个名字的出现次数
  .value();                // 获取链式操作的最终结果

console.log(resWithChain);
// 预期输出与上述相同

这种方法清晰地展示了Underscore.js链式调用的强大之处,它将数据转换和统计过程分解为易于理解的步骤。

深入理解:_.reduce()的正确使用与常见误区

虽然_.countBy()是解决此类问题的最佳选择,但理解如何正确使用_.reduce()也至关重要,尤其是在需要自定义聚合逻辑时。原始问题中尝试使用_.reduce()但遇到了问题,这暴露了一些JavaScript核心操作符和_.reduce()使用上的常见误区。

1. _.reduce()的常见误区分析

原先的_.reduce()尝试代码如下:

艾绘
艾绘

艾绘:一站式绘本创作平台,AI智能绘本设计神器!

下载
var firstNameOccurence = _.chain (nflTeams)
  .map(function(team) {return team.playersFirstNames})
  .flatten()
  .reduce(function(newObject, firstName){
      // 错误的逻辑
      return newObject[firstName] = 1 ? !newObject[firstName] :  newObject[firstName] += 1;
  }, {})
  .value();

这段代码的问题在于对JavaScript操作符(如赋值=、逻辑非!、三元运算符? :)的理解不准确,以及对_.reduce()回调函数返回值的作用不清晰。

  • 赋值操作符=的返回值: newObject[firstName] = 1这个表达式不仅会将1赋值给newObject[firstName],其整个表达式的值也是1。
  • 三元运算符的判断条件: newObject[firstName] = 1 ? ... : ... 这意味着三元运算符的条件部分始终是1,在布尔上下文中被视为true。因此,三元运算符的“真”分支将始终被执行。
  • 逻辑非!操作: !newObject[firstName]在newObject[firstName]被设置为1之后,!1会返回false。
  • _.reduce()回调的返回值: _.reduce()的回调函数必须返回累加器(accumulator)的下一个状态。在错误的逻辑中,它返回的是一个布尔值(false),而不是我们期望的对象。这意味着在第一次迭代后,newObject将不再是一个对象,而是一个布尔值false,后续操作将因尝试在非对象上设置属性而失败或产生意外结果。

因此,原始代码实际上会因为上述原因,在第一次迭代后返回false,并且在后续迭代中尝试在false上设置属性,导致最终结果不是一个对象,而是true或false。

2. _.reduce()的正确实现

要正确地使用_.reduce()实现统计功能,回调函数需要确保返回累加器对象,并且在对象中正确地更新计数。

const nflTeams = [
  { name: 'Kansas City Chiefs', playersFirstNames: ['Shane', 'Chad', 'Michael', 'Ronald', 'Blake', 'Noah'], champions: true },
  { name: 'Philadelphia Eagles', playersFirstNames: ['Jalen', 'Kenneth', 'Boston', 'Trey', 'Jack', 'Andre', 'Jack', 'Lane', 'Jason', 'Nakobe'], champions: false },
  { name: 'Cincinnati Bengals', playersFirstNames: ['Brandon', 'Joe', 'Chris', 'Joe', 'Tyler', 'Trenton', 'Trent', 'Mitchell', 'Alex', 'Trey', 'Ted'], champions: false },
  { name: 'San Francisco 49ers', playersFirstNames: ['Jimmy', 'Josh', 'Kyle', 'Jordan', 'Brandon', 'Danny', 'George', 'Tyler', 'Charlie', 'Jake', 'Nick', 'Nick', 'Kevin'], champions: false },
];

// 导入 Underscore.js 库
// 

const resWithReduce = _.chain(nflTeams)
  .map('playersFirstNames')
  .flatten()
  .reduce((currObject, firstName) => {
    // 如果 currObject[firstName] 不存在或为 0,则初始化为 0,然后加 1
    // 否则,直接在现有值上加 1
    currObject[firstName] = (currObject[firstName] || 0) + 1;
    return currObject; // 务必返回累加器对象
  }, {}) // 初始累加器是一个空对象
  .value();

console.log(resWithReduce);
// 预期输出与上述相同

代码解析:

  • _.chain(nflTeams).map('playersFirstNames').flatten():这部分与之前相同,用于获取所有扁平化的球员名字数组。
  • .reduce((currObject, firstName) => { ... }, {}):reduce方法接收一个回调函数和一个初始值({})。
    • currObject:当前累加器,即正在构建的统计对象。
    • firstName:当前迭代到的球员名字。
    • currObject[firstName] = (currObject[firstName] || 0) + 1;:这是核心逻辑。
      • currObject[firstName] || 0:如果currObject[firstName]已经存在(即该名字之前出现过),则使用其当前值;如果不存在(即该名字第一次出现),则将其视为0。
      • + 1:将获取到的值加1,实现计数。
    • return currObject;:关键一步! 每次迭代后,必须返回更新后的currObject,以便下一次迭代能基于最新的状态继续操作。

总结

在处理嵌套数组并统计元素频率的场景中,Underscore.js提供了多种强大且简洁的解决方案:

  1. 首选方案: 结合_.chain()、_.map()、_.flatten()和_.countBy()。这是最符合Underscore.js设计哲学且代码可读性高的方案。如果环境支持,也可以利用JavaScript原生的flatMap()进一步简化数据扁平化步骤。
  2. _.reduce()的替代方案: 尽管_.reduce()功能强大,但对于简单的计数任务,它不如_.countBy()直观。然而,理解_.reduce()的正确用法(特别是如何管理累加器和确保正确返回值)对于处理更复杂的聚合逻辑至关重要。

选择合适的工具和方法,不仅能提高代码效率,还能增强代码的可读性和可维护性。对于本教程中的问题,_.countBy()无疑是最高效和最优雅的解决方案。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
java基础知识汇总
java基础知识汇总

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

1501

2023.10.24

Go语言中的运算符有哪些
Go语言中的运算符有哪些

Go语言中的运算符有:1、加法运算符;2、减法运算符;3、乘法运算符;4、除法运算符;5、取余运算符;6、比较运算符;7、位运算符;8、按位与运算符;9、按位或运算符;10、按位异或运算符等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

232

2024.02.23

php三元运算符用法
php三元运算符用法

本专题整合了php三元运算符相关教程,阅读专题下面的文章了解更多详细内容。

87

2025.10.17

php三元运算符用法
php三元运算符用法

本专题整合了php三元运算符相关教程,阅读专题下面的文章了解更多详细内容。

87

2025.10.17

treenode的用法
treenode的用法

​在计算机编程领域,TreeNode是一种常见的数据结构,通常用于构建树形结构。在不同的编程语言中,TreeNode可能有不同的实现方式和用法,通常用于表示树的节点信息。更多关于treenode相关问题详情请看本专题下面的文章。php中文网欢迎大家前来学习。

538

2023.12.01

C++ 高效算法与数据结构
C++ 高效算法与数据结构

本专题讲解 C++ 中常用算法与数据结构的实现与优化,涵盖排序算法(快速排序、归并排序)、查找算法、图算法、动态规划、贪心算法等,并结合实际案例分析如何选择最优算法来提高程序效率。通过深入理解数据结构(链表、树、堆、哈希表等),帮助开发者提升 在复杂应用中的算法设计与性能优化能力。

17

2025.12.22

深入理解算法:高效算法与数据结构专题
深入理解算法:高效算法与数据结构专题

本专题专注于算法与数据结构的核心概念,适合想深入理解并提升编程能力的开发者。专题内容包括常见数据结构的实现与应用,如数组、链表、栈、队列、哈希表、树、图等;以及高效的排序算法、搜索算法、动态规划等经典算法。通过详细的讲解与复杂度分析,帮助开发者不仅能熟练运用这些基础知识,还能在实际编程中优化性能,提高代码的执行效率。本专题适合准备面试的开发者,也适合希望提高算法思维的编程爱好者。

27

2026.01.06

golang map内存释放
golang map内存释放

本专题整合了golang map内存相关教程,阅读专题下面的文章了解更多相关内容。

75

2025.09.05

俄罗斯Yandex引擎入口
俄罗斯Yandex引擎入口

2026年俄罗斯Yandex搜索引擎最新入口汇总,涵盖免登录、多语言支持、无广告视频播放及本地化服务等核心功能。阅读专题下面的文章了解更多详细内容。

158

2026.01.28

热门下载

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

精品课程

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

共58课时 | 4.3万人学习

TypeScript 教程
TypeScript 教程

共19课时 | 2.5万人学习

Bootstrap 5教程
Bootstrap 5教程

共46课时 | 3万人学习

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

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