在javascript中将对象数组中的相似键求和

时间:2018-10-09 11:31:16

标签: javascript arrays knockout.js grouping

我有一系列类似于以下内容的对象:-

0: {Id: 0, count1: 5, count2: 10, yearMonth: "201803"}
1: {Id: 0, count1: 10, count2: 0, yearMonth: "201804"}
2: {Id: 1, count1: 900, count2: 200, yearMonth: "201805"}
3: {Id: 0, count1: 10, count2: 0, yearMonth: "201806"}
4: {Id: 1, count1: 100, count2: 100, yearMonth: "201807"}
5: {Id: 1, count1: 100, count2: 2, yearMonth: "201808"}

我想对相似的密钥求和以获得 预期输出:

1: {Id: 0, count1: 25, count2: 10}
2: {Id: 1, count1: 1100, count2: 402}

是否可以使用任何数学运算或归约或任何其他javascript方法来实现?

谢谢

2 个答案:

答案 0 :(得分:1)

使用剔除,您可以使用一系列计算的属性来获得所需的(UI?)格式!

免责声明:我假设此列表将不包含数千个项目

1。分组

第一步是从项目列表(ko.observableArray([]))到按ID分组的计算对象:

// Search for "group by javascript" to have this function explained
const groupBy = (prop, xs) => xs.reduce(
  (acc, x) => Object.assign(acc, { [x[prop]]: (acc[x[prop]] || []).concat(x) }), {}
);

const items = ko.observableArray([]);

const itemsById = ko.pureComputed(() =>
  groupBy("Id", items())
);

itemsById.subscribe(console.log);
items([{Id: 0, count1: 5, count2: 10, yearMonth: "201803"},{Id: 0, count1: 10, count2: 0, yearMonth: "201804"},{Id: 1, count1: 900, count2: 200, yearMonth: "201805"},{Id: 0, count1: 10, count2: 0, yearMonth: "201806"},{Id: 1, count1: 100, count2: 100, yearMonth: "201807"},{Id: 1, count1: 100, count2: 2, yearMonth: "201808"}]);
<script src="https://cdnjs.cloudflare.com/ajax/libs/knockout/3.4.2/knockout-min.js"></script>

2。合并

现在,我们已经将需要汇总的项目列表进行了分组,我们可以开始应用合并逻辑:

const itemsWithSameId = [{Id:0,count1:5,count2:10,yearMonth:"201803"},{Id:0,count1:10,count2:0,yearMonth:"201804"},{Id:0,count1:10,count2:0,yearMonth:"201806"}];

const merge = (itemA, itemB) => ({
  Id: itemB.Id,
  count1: itemA.count1 + itemB.count1,
  count2: itemA.count2 + itemB.count2
});

// Look up "merging objects using reduce in javascript" to find out more
console.log(
  itemsWithSameId.reduce(merge, { count1: 0, count2: 0 })
)

3。从索引对象回到数组

现在我们知道如何合并我们的组了,我们可以移回用户界面中所需的数组:

// Utilities:
const groupBy = (prop, xs) => xs.reduce(
  (acc, x) => Object.assign(acc, {
    [x[prop]]: (acc[x[prop]] || []).concat(x)
  }), {}
);

// Data Logic:
const merge = (itemA, itemB) => ({
  Id: itemB.Id,
  count1: itemA.count1 + itemB.count1,
  count2: itemA.count2 + itemB.count2
});

// App
const items = ko.observableArray([]);
const itemsById = ko.pureComputed(() =>
  groupBy("Id", items())
);

// Look up "mapping over the values of a javascript object" for more info
const summedItems = ko.pureComputed(() =>
  Object
    .values(itemsById())
    .map(items => items.reduce(merge, { count1: 0, count2: 0 }))
);

// Apply bindings with viewmodel exposing summedItems
ko.applyBindings({ summedItems });

// Inject data (probably in success callback of ajax call)
items([{Id: 0, count1: 5, count2: 10, yearMonth: "201803"},{Id: 0, count1: 10, count2: 0, yearMonth: "201804"},{Id: 1, count1: 900, count2: 200, yearMonth: "201805"},{Id: 0, count1: 10, count2: 0, yearMonth: "201806"},{Id: 1, count1: 100, count2: 100, yearMonth: "201807"},{Id: 1, count1: 100, count2: 2, yearMonth: "201808"}]);
<script src="https://cdnjs.cloudflare.com/ajax/libs/knockout/3.4.2/knockout-min.js"></script>

<table>
  <thead>
    <tr>
      <th>Id</th>
      <th>Count 1</th>
      <th>Count 2</th>
    </tr>
  </thead>
  <tbody data-bind="foreach: summedItems">
    <td data-bind="text: Id"></td>
    <td data-bind="text: count1"></td>
    <td data-bind="text: count2"></td>
  </tbody>
</table>

答案 1 :(得分:0)

您可以通过使用哈希表和用于求和所需值的键数组来简化数组。

var data = [{ Id: 0, count1: 5, count2: 10, yearMonth: "201803" }, { Id: 0, count1: 10, count2: 0, yearMonth: "201804" }, { Id: 1, count1: 900, count2: 200, yearMonth: "201805" }, { Id: 0, count1: 10, count2: 0, yearMonth: "201806" }, { Id: 1, count1: 100, count2: 100, yearMonth: "201807" }, { Id: 1, count1: 100, count2: 2, yearMonth: "201808" }],
    keys = ['count1', 'count2'],
    grouped = Object.values(data.reduce((result, object) => {
        result[object.Id] = result[object.Id] || { Id: object.Id };
        keys.forEach(key => result[object.Id][key] = (result[object.Id][key] || 0) + object[key]);
        return result;
    }, Object.create(null)));

console.log(grouped);
.as-console-wrapper { max-height: 100% !important; top: 0; }

相关问题