Получить список дубликатов объектов в массиве объектов - PullRequest
0 голосов
/ 08 ноября 2018

Я пытаюсь получить дубликаты объектов в массиве объектов, скажем, объект похож на ниже.

  values = [
        { id:10; name: 'someName1' },
        { id:10; name: 'someName2' },
        { id:11; name: 'someName3' },
        { id:12; name: 'someName4' }
    ]

повторяющиеся объекты должны возвращаться как показано ниже

 duplicate= [
        { id:10; name: 'someName1' },
        { id:10; name: 'someName2' }
    ]

Ответы [ 8 ]

0 голосов
/ 30 марта 2019

Допустим, у вас есть:

arr = [
    { id:10, name: 'someName1' },
    { id:10, name: 'someName2' },
    { id:11, name: 'someName3' },
    { id:12, name: 'someName4' }
]

Итак, чтобы получить уникальные предметы:

unique = arr
     .map(e => e['id'])
     .map((e, i, final) => final.indexOf(e) === i && i)
     .filter(obj=> arr[obj])
     .map(e => arr[e]);

Тогда результат будет

unique = [
     { id:10, name: 'someName1' },
     { id:11, name: 'someName3' },
     { id:12, name: 'someName4' }
]

И, чтобы получить дубликаты идентификаторов:

duplicateIds = arr
     .map(e => e['id'])
     .map((e, i, final) => final.indexOf(e) !== i && i)
     .filter(obj=> arr[obj])
     .map(e => arr[e]["id"])

Список идентификаторов будет

duplicateIds = [10]

Таким образом, чтобы получить дубликаты объектов:

duplicate = arr.filter(obj=> dublicateIds.includes(obj.id));

Теперь у вас есть:

duplicate = [
    { id:10, name: 'someName1' },
    { id:10, name: 'someName2' }
]

Спасибо https://reactgo.com/removeduplicateobjects/

0 голосов
/ 08 ноября 2018

С lodash вы можете использовать _.groupBy() для группировки элементов по id. Чем _.filter() из групп, в которых менее двух членов, и _.flatten() результатов:

const values = [{id: 10, name: 'someName1'}, {id: 10, name: 'someName2'}, {id: 11, name:'someName3'}, {id: 12, name: 'someName4'}];

const result = _.flow([
  arr => _.groupBy(arr, 'id'), // group elements by id
  g => _.filter(g, o => o.length > 1), // remove groups that have less than two members
  _.flatten // flatten the results to a single array
])(values);

console.log(result);
<script src="https://cdnjs.cloudflare.com/ajax/libs/lodash.js/4.17.11/lodash.min.js"></script>
0 голосов
/ 09 ноября 2018

С помощью lodash вы можете решить эту проблему с помощью filter и countBy для сложности O(n):

const data = [{ id: 10,name: 'someName1' }, { id: 10,name: 'someName2' }, { id: 11,name: 'someName3' }, { id: 12,name: 'someName4' } ]

const counts = _.countBy(data, 'id')
console.log(_.filter(data, x => counts[x.id] > 1))
<script src="https://cdnjs.cloudflare.com/ajax/libs/lodash.js/4.17.10/lodash.min.js"></script>

Вы можете сделать то же самое с ES6 следующим образом:

const data = [{ id: 10,name: 'someName1' }, { id: 10,name: 'someName2' }, { id: 11,name: 'someName3' }, { id: 12,name: 'someName4' } ]

const countBy = (d, id) => d.reduce((r,{id},i,a) => (r[id] = a.filter(x => x.id == id).length, r),{})
const counts = countBy(data, 'id')
console.log(data.filter(x => [x.id] > 1))
0 голосов
/ 08 ноября 2018

Вы не выяснили, считаются ли два объекта с разными идентификаторами, но с одинаковым «именем», дубликатом. Я буду считать, что они не считаются дубликатами; другими словами, только объекты с одинаковым идентификатором будут считаться дубликатами.

let ids = {};
let dups = [];

values.forEach((val)=> {
  if (ids[val.id]) {
    // we have already found this same id
    dups.push(val)
  } else {
    ids[val.id] = true;
  }
})
return dups;
0 голосов
/ 08 ноября 2018

Вы можете использовать массив для хранения уникальных элементов и использовать фильтр по значениям, чтобы возвращать только дубликаты.

const unique = []

const duplicates = values.filter(o => {

   if(unique.find(i => i.id === o.id && i.name === o.name)) {
     return true
   }

   unique.push(o)
   return false;
})
0 голосов
/ 08 ноября 2018

Вы можете использовать reduce для создания таблицы поиска счетчиков на основе клавиши id и filter для удаления любых элементов, появившихся только один раз. Временная сложность O (n).

const values = [{id: 10, name: 'someName1'}, {id: 10, name: 'someName2'}, {id: 11, name:'someName3'}, {id: 12, name: 'someName4'}];

const lookup = values.reduce((a, e) => {
  a[e.id] = e.id in a ? ++a[e.id] : 0;
  return a;
}, {});

console.log(values.filter(e => lookup[e.id]));
0 голосов
/ 08 ноября 2018

создайте функцию, objectsEqual, а затем используйте фильтр

values.filter(function(num, elem) {
  for (var i = 0; i < values.length; i++) {
    if (i == num) continue;
    if (objectsEqual(values[num], values[i])) return true;
  }
  return false;
}
0 голосов
/ 08 ноября 2018

Попробуйте это

function checkDuplicateInObject(propertyName, inputArray) {
  var seenDuplicate = false,
  testObject = {};

  inputArray.map(function(item) {
  var itemPropertyName = item[propertyName];    
  if (itemPropertyName in testObject) {
  testObject[itemPropertyName].duplicate = true;
  item.duplicate = true;
  seenDuplicate = true;
 }
 else {
   testObject[itemPropertyName] = item;
   delete item.duplicate;
 }
});

 return seenDuplicate;
}

по ссылке: http://www.competa.com/blog/lets-find-duplicate-property-values-in-an-array-of-objects-in-javascript/

...