Combinei as respostas de eyelidlessness e KimKha.
A seguir, é um serviço angularjs e suporta números, seqüências de caracteres e objetos.
exports.Hash = () => {
let hashFunc;
function stringHash(string, noType) {
let hashString = string;
if (!noType) {
hashString = `string${string}`;
}
var hash = 0;
for (var i = 0; i < hashString.length; i++) {
var character = hashString.charCodeAt(i);
hash = ((hash<<5)-hash)+character;
hash = hash & hash; // Convert to 32bit integer
}
return hash;
}
function objectHash(obj, exclude) {
if (exclude.indexOf(obj) > -1) {
return undefined;
}
let hash = '';
const keys = Object.keys(obj).sort();
for (let index = 0; index < keys.length; index += 1) {
const key = keys[index];
const keyHash = hashFunc(key);
const attrHash = hashFunc(obj[key], exclude);
exclude.push(obj[key]);
hash += stringHash(`object${keyHash}${attrHash}`, true);
}
return stringHash(hash, true);
}
function Hash(unkType, exclude) {
let ex = exclude;
if (ex === undefined) {
ex = [];
}
if (!isNaN(unkType) && typeof unkType !== 'string') {
return unkType;
}
switch (typeof unkType) {
case 'object':
return objectHash(unkType, ex);
default:
return stringHash(String(unkType));
}
}
hashFunc = Hash;
return Hash;
};
Exemplo de uso:
Hash('hello world'), Hash('hello world') == Hash('hello world')
Hash({hello: 'hello world'}), Hash({hello: 'hello world'}) == Hash({hello: 'hello world'})
Hash({hello: 'hello world', goodbye: 'adios amigos'}), Hash({hello: 'hello world', goodbye: 'adios amigos'}) == Hash({goodbye: 'adios amigos', hello: 'hello world'})
Hash(['hello world']), Hash(['hello world']) == Hash(['hello world'])
Hash(1), Hash(1) == Hash(1)
Hash('1'), Hash('1') == Hash('1')
Resultado
432700947 true
-411117486 true
1725787021 true
-1585332251 true
1 true
-1881759168 true
Explicação
Como você pode ver, o coração do serviço é a função hash criada por KimKha. Adicionei tipos às seqüências de caracteres para que a estrutura do objeto também impactasse o valor final do hash. As chaves são hash para evitar colisões de objetos |
A comparação de objetos sem pálpebras é usada para evitar recursões infinitas por objetos auto-referenciados.
Uso
Eu criei este serviço para poder ter um serviço de erro que é acessado com objetos. Para que um serviço possa registrar um erro em um determinado objeto e outro possa determinar se foram encontrados erros.
ie
JsonValidation.js
ErrorSvc({id: 1, json: '{attr: "not-valid"}'}, 'Invalid Json Syntax - key not double quoted');
UserOfData.js
ErrorSvc({id: 1, json: '{attr: "not-valid"}'});
Isso retornaria:
['Invalid Json Syntax - key not double quoted']
Enquanto
ErrorSvc({id: 1, json: '{"attr": "not-valid"}'});
Isso retornaria
[]