我上面使用的 Set 只对原语有用。
这是不正确的,它适用于对象。问题是具有相同属性和属性值的不同对象不相等,因此执行两次set.add({"language": "ecmascript"}); 会将两个不相等的对象添加到集合中(都具有相同的属性名称和值)。
如果你多次添加 same 对象,它不会被添加第二次:
const set = new Set();
const obj = {"language": "ecmascript"};
set.add(obj);
set.add(obj);
console.log(set.size); // 1
Javascript 是否有内置类型用于...
如果您希望将具有相同属性和值的对象视为相等,那么不。您需要能够指定比较操作,并且 JavaScript 中没有内置的 Set 可以让您定义要使用的比较操作。
显然,您可以创建一个。作为一个起点,我可能会使用Map,它以对象上的属性名称为键,通过JSON.stringify 排序并转换为字符串。 (尽管如果您想将 Symbol 键作为相等定义的一部分,这将不起作用。)例如,如果您只考虑自己的属性:
const key = JSON.stringify(Object.getOwnPropertyNames(object).sort());
条目的值可能只是一个对象数组,其中包含您对其进行线性搜索的那些键,也可能是第二个Map,由某种属性值的哈希键控,具体取决于对象的数量你需要处理...
在cmets中,我问:
您只需要处理具有 JSON 可序列化值的对象吗?
你回答:
我有一堆已经序列化的对象,但有一些重复我想删除然后重新序列化。
是的,如果您不介意重新序列化,可以使用Set,如果您想跳过重新序列化部分,可以使用Map:
const unique = new Map();
for (const source of serializedObjects) {
const sourceObject = JSON.parse(source); // Or parse from whatever serialization it is
// Build object adding properties in alpha order for stability
const keyObj = {};
for (const key of Object.keys(sourceObject).sort()) {
keyObj[key] = sourceObject[key];
}
// Save it using JSON.stringify, which uses ES2015 property order
map.set(JSON.stringify(keyObj), source);
}
const uniqueSourceStrings = [...map.values()];
或者对于反序列化的对象本身:
const unique = new Map();
for (const source of serializedObjects) {
const sourceObject = JSON.parse(source); // Or parse from whatever serialization it is
// Build object adding properties in alpha order for stability
const keyObj = {};
for (const key of Object.keys(sourceObject).sort()) {
keyObj[key] = sourceObject[key];
}
// Save it using JSON.stringify, which uses ES2015 property order
map.set(JSON.stringify(keyObj), sourceObject); // <=================== changed
}
const uniqueSourceObject = [...map.values()];
// ^^================================================================== changed