Pergunta

Quero comparar 2 matrizes de objetos em código JavaScript.Os objetos têm 8 propriedades no total, mas cada objeto não terá um valor para cada um, e as matrizes nunca serão maiores que 8 itens cada, então talvez o método de força bruta de percorrer cada um e depois observar os valores do 8 propriedades é a maneira mais fácil de fazer o que quero, mas antes de implementar queria ver se alguém tinha uma solução mais elegante.Alguma ideia?

Foi útil?

Solução

EDITAR:Você não pode sobrecarregar operadores em implementações atuais e comuns de interpretadores JavaScript baseadas em navegador.

Para responder à pergunta original, uma maneira de fazer isso, e lembre-se, isso é um truque, simplesmente serializar os dois arrays para JSON e compare as duas strings JSON.Isso simplesmente lhe diria se as matrizes são diferentes, obviamente você poderia fazer isso para cada dos objetos dentro das matrizes também para ver quais eram diferentes.

Outra opção é usar uma biblioteca que tenha algumas facilidades interessantes para comparar objetos - eu uso e recomendo Kit Mochi.


EDITAR: A resposta que Kamens deu merece consideração também, já que uma única função para comparar dois objetos seria muito menor do que qualquer biblioteca para fazer o que sugiro (embora minha sugestão certamente funcionasse bem o suficiente).

Aqui está uma implementação ingênua que pode fazer apenas o suficiente para você - esteja ciente de que existem problemas potenciais com esta implementação:

function objectsAreSame(x, y) {
   var objectsAreSame = true;
   for(var propertyName in x) {
      if(x[propertyName] !== y[propertyName]) {
         objectsAreSame = false;
         break;
      }
   }
   return objectsAreSame;
}

A suposição é que ambos os objetos tenham exatamente a mesma lista de propriedades.

Ah, e provavelmente é óbvio que, para o bem ou para o mal, pertenço ao campo do único ponto de retorno.:)

Outras dicas

Eu sei que esta é uma pergunta antiga e as respostas fornecidas funcionam bem ...mas isso é um pouco mais curto e não requer bibliotecas adicionais (ou seja,JSON):

function arraysAreEqual(ary1,ary2){
  return (ary1.join('') == ary2.join(''));
}

Honestamente, com no máximo 8 objetos e no máximo 8 propriedades por objeto, sua melhor aposta é apenas percorrer cada objeto e fazer as comparações diretamente.Será rápido e será fácil.

Se você usar esses tipos de comparações com frequência, concordo com Jason sobre a serialização JSON... mas, caso contrário, não há necessidade de desacelerar seu aplicativo com uma nova biblioteca ou código de serialização JSON.

Trabalhei um pouco em um algoritmo simples para comparar o conteúdo de dois objetos e retornar uma lista inteligível de diferenças.Pensei em compartilhar.Ele empresta algumas ideias para jQuery, nomeadamente o map implementação de funções e verificação de tipo de objeto e array.

Ele retorna uma lista de "objetos diff", que são arrays com as informações diff.É muito simples.

Aqui está:

// 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;
}

tentei JSON.stringify() e trabalhou para mim.

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

Como a serialização geralmente não funciona (somente quando a ordem das propriedades corresponde: JSON.stringify({a:1,b:2}) !== JSON.stringify({b:2,a:1})) você deve verificar a contagem de propriedades e comparar cada propriedade também:

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

Se precisar de uma comparação profunda, você pode chamar a função recursivamente:

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

Então é fácil usar esta função para comparar objetos em arrays:

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

Tente este, por favor:

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;
  }
}

Aqui está minha tentativa, usando Módulo de afirmação do nó + pacote npm hash de objeto.

Suponho que você gostaria de verificar se dois arrays contêm os mesmos objetos, mesmo que esses objetos estejam ordenados de forma diferente entre os dois arrays.

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);
    });
}

O objectsAreSame função mencionada na resposta de @JasonBunting funciona bem para mim.No entanto, há um pequeno problema:Se x[propertyName] e y[propertyName] são objetos (typeof x[propertyName] == 'object'), você precisará chamar a função recursivamente para compará-las.

usando _.some de Lodash: 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);
Licenciado em: CC-BY-SA com atribuição
Não afiliado a StackOverflow
scroll top