JS中的groupBy方法

最近工作中需要写一些运营数据报表的页面,后端返回的数据都是未处理过的json,所以,大量的运算任务交给了前端来做,这其中有一个功能就是对数据进行分组,一开始我自己也尝试写了一些,但代码量比较大,显得很繁琐,直到后来在stackoverflow上看到了某位大牛写的方法,非常简洁优雅,代码如下:

function groupBy( array , f ) {
    let groups = {};
    array.forEach( function( o ) {
        let group = JSON.stringify( f(o) );
        groups[group] = groups[group] || [];
        groups[group].push( o );
    });
    return Object.keys(groups).map( function( group ) {
        return groups[group];
    });
}

let list = [
    {"name": "John","Average":15,"High":10,"DtmStamp":1358226000000},
    {"name": "Jane","Average":16,"High":92,"DtmStamp":1358226000000},
    {"name": "Jane","Average":17,"High":45,"DtmStamp":1358226000000},
    {"name": "John","Average":18,"High":87,"DtmStamp":1358226000000},
    {"name": "Jane","Average":15,"High":10,"DtmStamp":1358226060000},
    {"name": "John","Average":16,"High":87,"DtmStamp":1358226060000},
    {"name": "John","Average":17,"High":45,"DtmStamp":1358226060000},
    {"name": "Jane","Average":18,"High":92,"DtmStamp":1358226060000}
];

let sorted = groupBy(list, function(item){
    return [item.name];
});
console.log(sorted);

具体实现思路:

  1. 函数groupBy有两个形参,一为对象数组,二为匿名函数(该函数功能:返回对象的某个指定属性的属性值并存放在数组中);
  2. groupBy函数内,先创建一个空对象;
  3. 然后forEach遍历对象数组,遍历时要执行的函数中只有一个形参o(数组中的每个元素);
  4. 由于下面函数调用是想用name来分组,因此let group = JSON.stringify( f(o) ),相当于先获取到对象数组list中的name属性对应的属性值并放入数组中:["John"],然后再将属性值转换为json字符串:'["John"]';
  5. groups[group] = groups[group] || [],在js中对象也是关联数组,因此这里相当于做了两件事,一是把group作为groups的key,二是将对应的value初始化,第一次执行为空数组,循环执行时找到相同的name时保持不变;
  6. groups[group].push( o ),这句相当于把list中每个对象压入数组中作为value;
  7. 最后,Object.keys(groups)是取出groups对象中的所有key,然后遍历一个个key组成的新数组,返回分好了组的二维数组,至此大功告成~~~

结果如下:

[ [ { name: 'John', Average: 15, High: 10, DtmStamp: 1358226000000 },
    { name: 'John', Average: 18, High: 87, DtmStamp: 1358226000000 },
    { name: 'John', Average: 16, High: 87, DtmStamp: 1358226060000 },
    { name: 'John', Average: 17, High: 45, DtmStamp: 1358226060000 } ],
  [ { name: 'Jane', Average: 16, High: 92, DtmStamp: 1358226000000 },
    { name: 'Jane', Average: 17, High: 45, DtmStamp: 1358226000000 },
    { name: 'Jane', Average: 15, High: 10, DtmStamp: 1358226060000 },
    { name: 'Jane', Average: 18, High: 92, DtmStamp: 1358226060000 } ] ]
### JavaScript 实现类似于 SQL 的 GROUP BY 操作 在 JavaScript 中,虽然没有内置的 `GROUP BY` 方法像 SQL 那样直接提供分组功能,但可以通过组合数组方法来实现类似的逻辑。以下是通过 JavaScript 实现类似 SQL `GROUP BY` 功能的方法: #### 使用 `reduce()` 方法 可以利用 `Array.prototype.reduce()` 来创建一个对象或 Map 数据结构,其中键表示分组依据,值是一个包含该组所有项的数组。 ```javascript const data = [ { category: 'A', value: 10 }, { category: 'B', value: 20 }, { category: 'A', value: 30 }, { category: 'B', value: 40 } ]; // Group by 'category' const groupedData = data.reduce((accumulator, current) => { const key = current.category; if (!accumulator[key]) { accumulator[key] = []; } accumulator[key].push(current); return accumulator; }, {}); console.log(groupedData); ``` 上述代码会生成如下结果[^1]: ```json { "A": [{ "category": "A", "value": 10 }, { "category": "A", "value": 30 }], "B": [{ "category": "B", "value": 20 }, { "category": "B", "value": 40 }] } ``` 如果需要进一步聚合数据(例如求和),可以在每次迭代时更新累加器中的值。 ```javascript const aggregatedData = data.reduce((accumulator, current) => { const key = current.category; if (!accumulator[key]) { accumulator[key] = { category: key, totalValue: 0 }; } accumulator[key].totalValue += current.value; return accumulator; }, {}); console.log(Object.values(aggregatedData)); ``` 这将返回按类别汇总后的总值: ```json [ { "category": "A", "totalValue": 40 }, { "category": "B", "totalValue": 60 } ] ``` #### 使用第三方库 对于更复杂的场景,可以考虑使用 Lodash 或其他工具库提供的函数简化操作。Lodash 提供了一个名为 `_.groupBy` 的方法,可以直接用于分组。 ```javascript const _ = require('lodash'); const data = [ { category: 'A', value: 10 }, { category: 'B', value: 20 }, { category: 'A', value: 30 }, { category: 'B', value: 40 } ]; const groupedByCategory = _.groupBy(data, 'category'); console.log(groupedByCategory); ``` 此代码的结果与之前手动实现的方式相同。 --- #### 总结 尽管 JavaScript 并未原生支持 SQL 的 `GROUP BY` 关键字,但借助其强大的高阶函数(如 `reduce` 和 `map`)以及外部库(如 Lodash),能够轻松实现相似的功能并满足大多数需求。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值