如何比较对象数组中的对象值?

How to compare object values in an array of objects?

提问人:Tyler Morales 提问时间:5/27/2022 更新时间:5/27/2022 访问量:1172

问:

我正在尝试比较对象数组。每个对象都有相同的键,但每个键的值不同。我想创建一个函数来比较每个对象的相似键值对。

我只关心每个对象的质量和位置键,我想将所有对象与这两个键进行比较。

例如,如果对象数组中的第一个和第二个对象包含两个键的相同值,我想创建一个对象的输出数组来汇总每个分组。

解释: 对象 1 和 2 包含相同的值 和 。由于第三个对象没有,因此应创建一个新对象来汇总来自第一个和第二个对象的信息。该新对象应包含所有水果名称的数组和所有标签的数组。输出对象如下所示。qualitylocation

// Input
data = [
  {
    id: '1',
    fruit: 'granny smith',
    quality: 'good',
    location: 'chicago',
    tags: ['green', 'sweet'],
  },
  {
    id: '2',
    fruit: 'Fuji',
    quality: 'good',
    location: 'chicago',
    tags: ['red', 'tart'],
  },
  {
    id: '3',
    fruit: 'gala',
    quality: 'bad',
    location: 'san diego',
    tags: ['tall', 'thin'],
  },
];

// Function
function compareObjects(arr) {
  const grouped = [];

  // Loop over every fruit
  const similarObjects = arr.filter((obj, id) => {
    // create structure for each common object
    let shape = {
      id,
      fruits: [],
      quality: '',
      tags: [],
    };

    arr.forEach((item) => {
      // Return a new shape object that contains all fruit names, tags, and quality
      if (item.quality == obj.quality && item.location == obj.location) {
        shape.id = id;
        shape.fruits.push(item.fruit);
        shape.fruits.push(obj.fruit);
        shape.quality = item.quality;
        shape.tags.push(item.tag);
        shape.tags.push(obj.tag);
        return shape;
      }
    });
    return obj;
  });

  return similarObjects;
}

console.log(compareObjects(data));

// Output
const output = [
  {
    id: 'a',
    fruits: ['grann smith', 'fuji'],
    quality: 'good',
    tags: ['green', 'sweet', 'red', 'tart'],
  },
  ...
];


JavaScript 数组 算法 对象 比较

评论

0赞 SSM 5/27/2022
第三个对象是否出现在 ?从哪里来?outputid"a"
1赞 Tyler Morales 5/27/2022
第三个对象不应出现在输出中,因为它与其他对象的任何键值对都不匹配。“A”只是一个唯一生成的 ID。它在这方面没有多大用处。

答:

1赞 SSM 5/27/2022 #1

您可以使用 Array.prototype.reduce 按 它们 和 进行分组,并过滤长度大于 1 的组。dataqualitylocation

const 
  data = [
    { id: "1", fruit: "granny smith", quality: "good", location: "chicago", tags: ["green", "sweet"] },
    { id: "2", fruit: "Fuji", quality: "good", location: "chicago", tags: ["red", "tart"] },
    { id: "3", fruit: "gala", quality: "bad", location: "san diego", tags: ["tall", "thin"] },
  ],
  output = Object.values(
    data.reduce((r, d) => {
      const key = `${d.quality}+${d.location}`;
      if (!r[key]) {
        r[key] = { id: d.id, fruits: [], quality: d.quality, location: d.location, tags: [] };
      }
      r[key].fruits.push(d.fruit);
      r[key].tags.push(...d.tags);
      return r;
    }, {})
  ).filter((d) => d.fruits.length > 1);

console.log(output);

如果您还希望只保留唯一的水果,那么您可以映射到结果数组并使用 Set 删除重复项。

const 
  data = [
    { id: "1", fruit: "granny smith", quality: "good", location: "chicago", tags: ["green", "sweet"] },
    { id: "2", fruit: "Fuji", quality: "good", location: "chicago", tags: ["red", "tart"] },
    { id: "3", fruit: "gala", quality: "bad", location: "san diego", tags: ["tall", "thin"] },
  ],
  output = Object.values(
    data.reduce((r, d) => {
      const key = `${d.quality}+${d.location}`;
      if (!r[key]) {
        r[key] = { id: d.id, fruits: [], quality: d.quality, location: d.location, tags: [] };
      }
      r[key].fruits.push(d.fruit);
      r[key].tags.push(...d.tags);
      return r;
    }, {})
  )
    .filter((d) => d.fruits.length > 1)
    .map((d) => ({ ...d, fruits: [...new Set(d.fruits)] }));

console.log(output);

其他相关文件:

评论

0赞 Tyler Morales 5/27/2022
这看起来不错。你能在输出中组合来自两个对象的标签吗?标签输出应如下所示: tags: [ 'red', 'tart', 'green', 'sweet' ]
1赞 SSM 5/27/2022
哦,是的,固定了!
1赞 Tyler Morales 5/27/2022
我看到你在那里做了什么。您只需将两个数组展开即可。伟大!
0赞 Tyler Morales 5/27/2022
还有一件快速的事情。我正在尝试删除水果名称数组中重复的水果名称。例如,如果两个对象的水果名称都是“granny smith”,则输出 fruits 数组应只包含“granny smith”,一次,而不是两次。
1赞 SSM 5/27/2022
@TylerMorales没错,您也可以在最后通过链接呼叫一次完成所有操作,我已经更新了我的答案。.map