Сравнение массивов объектов в JavaScript
-
09-06-2019 - |
Вопрос
Я хочу сравнить 2 массива объектов в коде JavaScript.Объекты имеют всего 8 свойств, но каждый объект не будет иметь значения для каждого, и массивы никогда не будут больше 8 элементов каждый, так что, возможно, метод перебора каждого из них с последующим просмотром значений 8 свойств - самый простой способ сделать то, что я хочу сделать, но перед реализацией я хотел посмотреть, есть ли у кого-нибудь более элегантное решение.Есть какие-нибудь мысли?
Решение
Редактировать:Вы не можете перегружать операторы в текущих, распространенных браузерных реализациях интерпретаторов JavaScript.
Чтобы ответить на первоначальный вопрос, есть один способ, которым вы могли бы это сделать, и имейте в виду, это своего рода взлом, просто сериализуйте два массива в JSON а затем сравните две строки JSON.Это просто сообщило бы вам, отличаются ли массивы, очевидно, вы могли бы сделать это, чтобы каждый объектов внутри массивов, а также, чтобы увидеть, какие из них отличались друг от друга.
Другой вариант - использовать библиотеку, в которой есть несколько приятных возможностей для сравнения объектов - я использую и рекомендую Мочикит.
Редактировать: Ответ , который дал каменс заслуживает рассмотрения также, поскольку одна функция для сравнения двух заданных объектов была бы намного меньше, чем любая библиотека для выполнения того, что я предлагаю (хотя мое предложение, безусловно, сработало бы достаточно хорошо).
Вот наивная реализация, которой может быть достаточно для вас - имейте в виду, что с этой реализацией могут возникнуть потенциальные проблемы:
function objectsAreSame(x, y) {
var objectsAreSame = true;
for(var propertyName in x) {
if(x[propertyName] !== y[propertyName]) {
objectsAreSame = false;
break;
}
}
return objectsAreSame;
}
Предполагается, что оба объекта имеют один и тот же точный список свойств.
Да, и, наверное, очевидно, что, к лучшему это или к худшему, я принадлежу к лагерю сторонников только одной точки возврата.:)
Другие советы
Я знаю, что это старый вопрос, и предоставленные ответы работают нормально...но это немного короче и не требует каких-либо дополнительных библиотек ( т. е.JSON ):
function arraysAreEqual(ary1,ary2){
return (ary1.join('') == ary2.join(''));
}
Честно говоря, при максимальном количестве объектов 8 и максимальном количестве свойств 8 для каждого объекта лучше всего просто обойти каждый объект и провести сравнение напрямую.Это будет быстро и легко.
Если вы собираетесь часто использовать эти типы сравнений, то я согласен с Джейсоном по поводу сериализации в формате JSON...но в остальном нет необходимости замедлять работу вашего приложения с помощью новой библиотеки или кода сериализации в формате JSON.
Я немного поработал над простым алгоритмом сравнения содержимого двух объектов и возврата понятного списка различий.Подумал, что стоит поделиться.Он заимствует некоторые идеи для jQuery, а именно map
реализация функции и проверка типа объекта и массива.
Он возвращает список "объектов diff", которые представляют собой массивы с информацией о различии.Все очень просто.
Вот оно:
// compare contents of two objects and return a list of differences
// returns an array where each element is also an array in the form:
// [accessor, diffType, leftValue, rightValue ]
//
// diffType is one of the following:
// value: when primitive values at that index are different
// undefined: when values in that index exist in one object but don't in
// another; one of the values is always undefined
// null: when a value in that index is null or undefined; values are
// expressed as boolean values, indicated wheter they were nulls
// type: when values in that index are of different types; values are
// expressed as types
// length: when arrays in that index are of different length; values are
// the lengths of the arrays
//
function DiffObjects(o1, o2) {
// choose a map() impl.
// you may use $.map from jQuery if you wish
var map = Array.prototype.map?
function(a) { return Array.prototype.map.apply(a, Array.prototype.slice.call(arguments, 1)); } :
function(a, f) {
var ret = new Array(a.length), value;
for ( var i = 0, length = a.length; i < length; i++ )
ret[i] = f(a[i], i);
return ret.concat();
};
// shorthand for push impl.
var push = Array.prototype.push;
// check for null/undefined values
if ((o1 == null) || (o2 == null)) {
if (o1 != o2)
return [["", "null", o1!=null, o2!=null]];
return undefined; // both null
}
// compare types
if ((o1.constructor != o2.constructor) ||
(typeof o1 != typeof o2)) {
return [["", "type", Object.prototype.toString.call(o1), Object.prototype.toString.call(o2) ]]; // different type
}
// compare arrays
if (Object.prototype.toString.call(o1) == "[object Array]") {
if (o1.length != o2.length) {
return [["", "length", o1.length, o2.length]]; // different length
}
var diff =[];
for (var i=0; i<o1.length; i++) {
// per element nested diff
var innerDiff = DiffObjects(o1[i], o2[i]);
if (innerDiff) { // o1[i] != o2[i]
// merge diff array into parent's while including parent object name ([i])
push.apply(diff, map(innerDiff, function(o, j) { o[0]="[" + i + "]" + o[0]; return o; }));
}
}
// if any differences were found, return them
if (diff.length)
return diff;
// return nothing if arrays equal
return undefined;
}
// compare object trees
if (Object.prototype.toString.call(o1) == "[object Object]") {
var diff =[];
// check all props in o1
for (var prop in o1) {
// the double check in o1 is because in V8 objects remember keys set to undefined
if ((typeof o2[prop] == "undefined") && (typeof o1[prop] != "undefined")) {
// prop exists in o1 but not in o2
diff.push(["[" + prop + "]", "undefined", o1[prop], undefined]); // prop exists in o1 but not in o2
}
else {
// per element nested diff
var innerDiff = DiffObjects(o1[prop], o2[prop]);
if (innerDiff) { // o1[prop] != o2[prop]
// merge diff array into parent's while including parent object name ([prop])
push.apply(diff, map(innerDiff, function(o, j) { o[0]="[" + prop + "]" + o[0]; return o; }));
}
}
}
for (var prop in o2) {
// the double check in o2 is because in V8 objects remember keys set to undefined
if ((typeof o1[prop] == "undefined") && (typeof o2[prop] != "undefined")) {
// prop exists in o2 but not in o1
diff.push(["[" + prop + "]", "undefined", undefined, o2[prop]]); // prop exists in o2 but not in o1
}
}
// if any differences were found, return them
if (diff.length)
return diff;
// return nothing if objects equal
return undefined;
}
// if same type and not null or objects or arrays
// perform primitive value comparison
if (o1 != o2)
return [["", "value", o1, o2]];
// return nothing if values are equal
return undefined;
}
Я пытался JSON.stringify()
и сработало на меня.
let array1 = [1,2,{value:'alpha'}] , array2 = [{value:'alpha'},'music',3,4];
JSON.stringify(array1) // "[1,2,{"value":"alpha"}]"
JSON.stringify(array2) // "[{"value":"alpha"},"music",3,4]"
JSON.stringify(array1) === JSON.stringify(array2); // false
Поскольку сериализация обычно не работает (только когда порядок свойств совпадает: JSON.stringify({a:1,b:2}) !== JSON.stringify({b:2,a:1})
) вы должны проверить количество свойств, а также сравнить каждое свойство:
const objectsEqual = (o1, o2) =>
Object.keys(o1).length === Object.keys(o2).length
&& Object.keys(o1).every(p => o1[p] === o2[p]);
const obj1 = { name: 'John', age: 33};
const obj2 = { age: 33, name: 'John' };
const obj3 = { name: 'John', age: 45 };
console.log(objectsEqual(obj1, obj2)); // true
console.log(objectsEqual(obj1, obj3)); // false
Если вам нужно глубокое сравнение, вы можете вызвать функцию рекурсивно:
const obj1 = { name: 'John', age: 33, info: { married: true, hobbies: ['sport', 'art'] } };
const obj2 = { age: 33, name: 'John', info: { hobbies: ['sport', 'art'], married: true } };
const obj3 = { name: 'John', age: 33 };
const objectsEqual = (o1, o2) =>
typeof o1 === 'object' && Object.keys(o1).length > 0
? Object.keys(o1).length === Object.keys(o2).length
&& Object.keys(o1).every(p => objectsEqual(o1[p], o2[p]))
: o1 === o2;
console.log(objectsEqual(obj1, obj2)); // true
console.log(objectsEqual(obj1, obj3)); // false
Тогда эту функцию легко использовать для сравнения объектов в массивах:
const arr1 = [obj1, obj1];
const arr2 = [obj1, obj2];
const arr3 = [obj1, obj3];
const arraysEqual = (a1, a2) =>
a1.length === a2.length && a1.every((o, idx) => objectsEqual(o, a2[idx]));
console.log(arraysEqual(arr1, arr2)); // true
console.log(arraysEqual(arr1, arr3)); // false
Пожалуйста, попробуйте вот это:
function used_to_compare_two_arrays(a, b)
{
// This block will make the array of indexed that array b contains a elements
var c = a.filter(function(value, index, obj) {
return b.indexOf(value) > -1;
});
// This is used for making comparison that both have same length if no condition go wrong
if (c.length !== a.length) {
return 0;
} else{
return 1;
}
}
Вот моя попытка, используя Модуль assert узла + пакет npm объект-хэш.
Я полагаю, что вы хотели бы проверить, содержат ли два массива одни и те же объекты, даже если эти объекты упорядочены по-разному между двумя массивами.
var assert = require('assert');
var hash = require('object-hash');
var obj1 = {a: 1, b: 2, c: 333},
obj2 = {b: 2, a: 1, c: 444},
obj3 = {b: "AAA", c: 555},
obj4 = {c: 555, b: "AAA"};
var array1 = [obj1, obj2, obj3, obj4];
var array2 = [obj3, obj2, obj4, obj1]; // [obj3, obj3, obj2, obj1] should work as well
// calling assert.deepEquals(array1, array2) at this point FAILS (throws an AssertionError)
// even if array1 and array2 contain the same objects in different order,
// because array1[0].c !== array2[0].c
// sort objects in arrays by their hashes, so that if the arrays are identical,
// their objects can be compared in the same order, one by one
var array1 = sortArrayOnHash(array1);
var array2 = sortArrayOnHash(array2);
// then, this should output "PASS"
try {
assert.deepEqual(array1, array2);
console.log("PASS");
} catch (e) {
console.log("FAIL");
console.log(e);
}
// You could define as well something like Array.prototype.sortOnHash()...
function sortArrayOnHash(array) {
return array.sort(function(a, b) {
return hash(a) > hash(b);
});
}
Тот Самый objectsAreSame
функция, упомянутая в ответе @JasonBunting, у меня работает нормально.Однако есть небольшая проблема:Если x[propertyName]
и y[propertyName]
являются объектами (typeof x[propertyName] == 'object'
), вам нужно будет вызвать функцию рекурсивно, чтобы сравнить их.
используя _.some
из лодаша: https://lodash.com/docs/4.17.11#some
const array1AndArray2NotEqual =
_.some(array1, (a1, idx) => a1.key1 !== array2[idx].key1
|| a1.key2 !== array2[idx].key2
|| a1.key3 !== array2[idx].key3);