文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

使用 JavaScript 进行数据分组最优雅的方式

2024-12-02 11:29

关注

大家好,我是 ConardLi ,今天我们一起来看一个数据分组的小技巧。

对数据进行分组,是我们在开发中经常会遇到的需求,使用 JavaScript 进行数据分组的方式也有很多种,但是由于没有原生方法的支持,我们自己实现的数据分组函数通常都比较冗长而且难以理解。

不过,告诉大家一个好消息,一个专门用来做数据分组的提案 Array.prototype.groupBy 已经到达 Stage 3 啦!

在看这个提案,之前,我们先来回顾下我们以前在 JavaScript 里是怎么分组的。

以前的方式

假设我们有下面一组数据:

  1. const items = [ 
  2.   { 
  3.     type: 'clothes', 
  4.     value: '👔', 
  5.   }, 
  6.   { 
  7.     type: 'clothes', 
  8.     value: '👕', 
  9.   }, 
  10.   { 
  11.     type: 'clothes', 
  12.     value: '👗', 
  13.   }, 
  14.   { 
  15.     type: 'animal', 
  16.     value: '🐷', 
  17.   }, 
  18.   { 
  19.     type: 'animal', 
  20.     value: '🐸', 
  21.   }, 
  22.   { 
  23.     type: 'animal', 
  24.     value: '🐒', 
  25.   }, 
  26. ]; 

我们希望按照 type 分组成下面的格式:

  1. const items = { 
  2.   clothes: [ 
  3.     { 
  4.       type: 'clothes', 
  5.       value: '👔', 
  6.     }, 
  7.     { 
  8.       type: 'clothes', 
  9.       value: '👕', 
  10.     }, 
  11.     { 
  12.       type: 'clothes', 
  13.       value: '👗', 
  14.     }, 
  15.   ], 
  16.   animal: [ 
  17.     { 
  18.       type: 'animal', 
  19.       value: '🐷', 
  20.     }, 
  21.     { 
  22.       type: 'animal', 
  23.       value: '🐸', 
  24.     }, 
  25.     { 
  26.       type: 'animal', 
  27.       value: '🐒', 
  28.     }, 
  29.   ], 
  30. }; 

我们可能会用到下面的写法:

for 循环

最直接而且容易理解的方法,就是代码有点多。

  1. const groupedBy = {}; 
  2.  
  3. for (const item of items) { 
  4.   if (groupedBy[item.type]) { 
  5.     groupedBy[item.type].push(item); 
  6.   } else { 
  7.     groupedBy[item.type] = [item]; 
  8.   } 

reduce

使用 Array.protoype.reduce 虽然语法看起来简单,但是太难读了。

  1. items.reduce( 
  2.   (acc, item) => ({ 
  3.     ...acc, 
  4.     [item.type]: [...(acc[item.type] ?? []), item], 
  5.   }), 
  6.   {}, 
  7. ); 

我们稍微改造的容易理解一点,语法就跟上面的 for 循环差不多了:

  1. items.reduce((acc, item) => { 
  2.   if (acc[item.type]) { 
  3.     acc[item.type].push(item); 
  4.   } else { 
  5.     acc[item.type] = [item]; 
  6.   } 
  7.  
  8.   return acc; 
  9. }, {}); 

filter

使用 Array.prototype.filter,代码看起来很容易阅读,但是性能很差,你需要对数组进行多次过滤,而且如果 type 属性值比较多的情况下,还需要做更多的 filter 操作。

  1. const groupedBy = { 
  2.   fruit: items.filter((item) => item.type === 'clothes'), 
  3.   vegetable: items.filter((item) => item.type === 'animal'), 
  4. }; 

其他

如果你既不想用 reduce,还想用到函数式写法,你可能会写出下面的代码:

  1. Object.fromEntries( 
  2.   Array.from(new Set(items.map(({ type }) => type))).map((type) => [ 
  3.     type, 
  4.     items.filter((item) => item.type === type), 
  5.   ]), 
  6. ); 

是不是很让人崩溃 🤯~

Array.prototype.groupBy

好了,如果使用 Array.prototype.groupBy,你只需要下面这一行代码:

  1. items.groupBy(({ type }) => type); 

groupBy 的回调中一共有三个参数:

  1. const array = [1, 2, 3, 4, 5]; 
  2.  
  3. // groupBy groups items by arbitrary key. 
  4. // In this case, we're grouping by even/odd keys 
  5. array.groupBy((num, index, array) => { 
  6.   return num % 2 === 0 ? 'even': 'odd'; 
  7. }); 

另外,你还可以用 groupByToMap,将数据分组为一个 Map 对象。

  1. // groupByToMap returns items in a Map, and is useful for grouping using 
  2. // an object key. 
  3. const odd  = { odd: true }; 
  4. const even = { even: true }; 
  5. array.groupByToMap((num, index, array) => { 
  6.   return num % 2 === 0 ? even: odd; 
  7. }); 
  8.  
  9. // =>  Map { {odd: true}: [1, 3, 5], {even: true}: [2, 4] } 

 

来源:code秘密花园内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯