JavaScript數(shù)組中相同的元素進行分組(數(shù)據(jù)聚合)groupBy函數(shù)詳解
前言
今天在打算從js端時序數(shù)據(jù)庫TSDB中,按相同的類型的數(shù)據(jù)排在一起,并且取同一時間段最新的數(shù)據(jù),經(jīng)過查詢這種思想叫做數(shù)據(jù)聚合,就是返回的數(shù)據(jù)要根據(jù)一個屬性來做計算。
一、數(shù)據(jù)聚合
1.groupBy()函數(shù)
聚合用groupBy這個函數(shù),它傳兩個形參一為對象數(shù)組,二為匿名函數(shù)(該函數(shù)功能:返回對象的某個指定屬性的屬性值并存放在數(shù)組中)
groupBy: function(array, func){ var groups = {}; array.forEach(function(element){ var group = JSON.stringify(func(element)); groups[group] = groups[group] || []; //第一次執(zhí)行為空數(shù)組 //groups[group] = [] //這樣寫就會覆蓋之前的數(shù)據(jù),視情況而用 groups[group].push([element,group]);//這里可以單獨傳element,本人是需要用到group才需要進行push }); return Object.keys(groups).map(function(group){ return groups[group]; }); },
這里返回的是數(shù)組,需要返回key所對應(yīng)的數(shù)組可以
return Object.keys(groups).map(function(group){ return groups[group]; });
Object.keys(groups)是取出groups對象中的所有key,然后遍歷一個個key組成的新數(shù)組,返回分好了組的二維數(shù)組
- .groupBy函數(shù)內(nèi),先創(chuàng)建一個空對象;
- 然后forEach遍歷對象數(shù)組,遍歷時要執(zhí)行的函數(shù)中只有一個形參element(數(shù)組中的每個元素);
- 由于下面函數(shù)調(diào)用是想用name來分組,因此let group = JSON.stringify( func(element) ),相當于先獲取到數(shù)組中的想要用來排序的屬性(這里我用返回的new Date(item.timestamp * 1000).getHours(),就是時間戳轉(zhuǎn)換出的小時)對應(yīng)的屬性值并放入數(shù)組中,然后再將屬性值轉(zhuǎn)換為json字符串;
- groups[group] = groups[group] || [],在js中對象也是關(guān)聯(lián)數(shù)組,因此這里相當于做了兩件事,一是把group作為groups的key,二是將對應(yīng)的value初始化,第一次執(zhí)行為空數(shù)組,循環(huán)執(zhí)行時找到相同的name時保持不變;
- groups[group] = groups[group] || [],這里也可以寫成groups[group] = [],這樣的話每次push進去的數(shù)據(jù)就會覆蓋之前的數(shù)據(jù),從而達到每次都能獲取同一時間段內(nèi)的最新數(shù)據(jù)。
- groups[group].push([element,group]),這句相當于把list中每個對象壓入數(shù)組中作為value;
2.sortData()函數(shù)
寫好groupBy
函數(shù)后就可以調(diào)用進行排序了:
sortData: function(Data){ var sorted = this.groupBy(Data, function(item){ return new Date(item.timestamp * 1000).getHours(); // 返回需要分組的對象 // return item.name; // 返回需要分組的對象 }); return sorted; },
這里的return可以返回數(shù)組中參數(shù)進行比較,比如item中name屬性。item是groupBy
函數(shù)func(element)中的element
二、ECS6箭頭函數(shù)寫法
上面是ECS5的寫法,到了ECS6之后JS就引入箭頭函數(shù),下面是ECS6含箭頭函數(shù)的寫法。
1.sortClass()函數(shù)
sortClass(Data){ const groupBy = (array, func) => { let groups = {}; array.forEach((element)=>{ let group = JSON.stringify(func(element)); groups[group] = groups[group] || []; groups[group].push(element); }); return Object.keys(groups).map((group) =>{ return groups[group]; }) }; const sorted = groupBy(Data, (item) =>{ return item.name; }) return sorted; }
2.運行結(jié)果
這就是運行sortClass(res.query)之后得到的結(jié)果,這里本人是timestamp時間戳來進行排序,而紅框中的數(shù)據(jù)就是push[element,group]中的group。
三、按需聚合(結(jié)合實際使用)
上面是對數(shù)據(jù)庫中數(shù)據(jù)聚合得到是數(shù)組每個key對于都是當前最新的值(只有一個),如果這里要求要保留每個key都對于兩個最新的值,則需要對groupBy函數(shù)進行改動一下了。
1.groupBy()函數(shù)
/** * author: CSH * 函數(shù)功能:給定參數(shù)對數(shù)組進行分組并排序 * @param {Object} array [需要排序的數(shù)組] * @param {Object} func [該函數(shù)參數(shù)用于返回用于分組排序的對象] */ groupBy: function(array, func){ var groups = {}; array.forEach(function(element){ var group = JSON.stringify(func(element)); groups[group] = groups[group] || []; //第一次執(zhí)行為空數(shù)組 // groups[group] = []; // 這樣就能得到當前小時時間段內(nèi)最新的數(shù)據(jù) groups[group].push([element,group]); groups[group].length >= 3 ? groups[group].shift() : null; // 只保留當前數(shù)組最新兩組數(shù)據(jù) }); return Object.keys(groups).map(function(group){ return groups[group]; }); },
在js中對數(shù)組只保留最新兩項數(shù)據(jù)需要用到JS中數(shù)組里的 shift()方法 (位移元素)了,位移與彈出等同,但處理首個元素而不是最后一個。shift() 方法會刪除首個數(shù)組元素,并把所有其他元素“位移”到更低的索引了(類似于隊列一樣)。使用該方法就可以保證數(shù)組內(nèi)的元素都是最新兩項。
2.sortData()函數(shù)(無改動)
/** * author: CSH * 函數(shù)功能:給定數(shù)組,并返回需要分組的對象,然后根據(jù)該對象進行分組排序返回新的數(shù)組 * @param {Object} sortData [需要數(shù)據(jù)聚合的數(shù)組] */ sortClass: function(sortData){ var sorted = this.groupBy(sortData, function(item){ return new Date(item.timestamp * 1000).getHours(); // 返回需要分組的對象 // return item.name; // 返回需要分組的對象 }); return sorted; },
對groupBy()函數(shù)進行改動之后,在用sortData進行調(diào)用返回就能得到想要的數(shù)據(jù)了。
總結(jié)
以上就是今天要講的內(nèi)容,本文僅僅簡單介紹了JS數(shù)據(jù)聚合的方法,ECS5和ECS6分別都有不同的寫法。
到此這篇關(guān)于JavaScript數(shù)組中相同的元素進行分組(數(shù)據(jù)聚合) groupBy函數(shù)的文章就介紹到這了,更多相關(guān)js數(shù)組分組groupBy函數(shù)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
JavaScript實現(xiàn)簡單版的留言發(fā)布與刪除
這篇文章主要介紹了如何通過JavaScript實現(xiàn)簡單的留言板功能:留言的發(fā)布與刪除。文中的示例代碼講解詳細,感興趣的小伙伴可以學(xué)習(xí)一下2022-03-03利用javascript如何隨機生成一定位數(shù)的密碼
這篇文章主要給大家介紹了關(guān)于利用javascript如何隨機生成一定位數(shù)的密碼的相關(guān)資料,文中給出了詳細的示例代碼,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧。2017-09-09用js來定義瀏覽器中一個左右浮動元素相對于頁面主體寬度的位置的函數(shù)
用js來定義瀏覽器中一個左右浮動元素相對于頁面主體寬度的位置的函數(shù) 函數(shù)的參數(shù):elem元素、頁面主體寬度(如:990、950寬)、elem與頁面主體DIV的左邊距2012-01-01JavaScript仿京東實現(xiàn)秒殺倒計時案例詳解
這篇文章主要為大家詳細介紹了如何利用JavaScript實現(xiàn)京東秒殺倒計時效果,文中示例代碼介紹的非常詳細,感興趣的小伙伴們可以參考一下2022-03-03

JavaScript面向?qū)ο缶幊绦∮螒?--貪吃蛇代碼實例