最新消息:雨落星辰是一个专注网站SEO优化、网站SEO诊断、搜索引擎研究、网络营销推广、网站策划运营及站长类的自媒体原创博客

javascript - How to get deep-equality of keys in ES6 Map? Alternative to using complex object as ES6 Map key? - Stack Overflow

programmeradmin2浏览0评论

Here is some example Javascript (ES6) code that does not do what one might intuitively imagine.

const exampleMap = new Map([[{a: 1}, 2]]);
console.log(exampleMap.get({a: 1}));

As it turns out, this prints undefined. Why? The reasoning is covered in this StackOverflow answer. Per the MDN entry for Map, Map uses === for key equality. And, per the MDN entry for ===, Objects are pared by reference equality.

That's all fine and good. It does exactly what the docs say it should. Unfortunately, what the above code is trying to do would be quite useful, even if it isn't the actual behavior per the spec.

How can I use Map, or what should I use instead of Map, to get a key-value lookup where the keys are pared by object deep-equality semantics?

Here is some example Javascript (ES6) code that does not do what one might intuitively imagine.

const exampleMap = new Map([[{a: 1}, 2]]);
console.log(exampleMap.get({a: 1}));

As it turns out, this prints undefined. Why? The reasoning is covered in this StackOverflow answer. Per the MDN entry for Map, Map uses === for key equality. And, per the MDN entry for ===, Objects are pared by reference equality.

That's all fine and good. It does exactly what the docs say it should. Unfortunately, what the above code is trying to do would be quite useful, even if it isn't the actual behavior per the spec.

How can I use Map, or what should I use instead of Map, to get a key-value lookup where the keys are pared by object deep-equality semantics?

Share Improve this question asked Jul 29, 2019 at 22:58 MingMing 1,69314 silver badges27 bronze badges 2
  • "object deep-equality semantics"? You mean stringification? Then stringify it, and a simple plain Object will do. {a:1} is semantically different than {a:1} in javascript. – Kaiido Commented Jul 30, 2019 at 1:31
  • See also Define a custom hash() method for use with ES6 Maps – Bergi Commented Jun 13, 2021 at 23:35
Add a ment  | 

4 Answers 4

Reset to default 1

You could create your own function, where you pass through the map (m) you want to get the value of, and a key you want to search in the map. You can then stringify your key, such that you can search the keys in your map and pare them against this stringified key

const exampleMap = new Map([[{a: 1}, 2]]);

const getMapVal = (m, key) => {
  const strKey = JSON.stringify(key);
  return m.get(Array.from(m.keys()).find((k) => JSON.stringify(k) === strKey));
}

console.log(getMapVal(exampleMap, {a: 1})); // 2 (gives undefined if key doesn't exists)

Here's something that does a deep equality check. It relies on JSON.stringify though so it might be a performance issue for very large object keys. This also updates the Map prototype for use in the same way as .get.

Object.defineProperty(Map.prototype, 'deepCheck', {
  value: function(lookupKey) {
    let lookupValue;
    const lookupKeyStr = JSON.stringify(lookupKey);
    if (this == null) {
      throw new TypeError('this is null or not defined');
    }

    const iterator = this.entries();
    let result = null;


    while (true) {
      result = iterator.next();
      if (result.done) {
        break;
      }
      if (JSON.stringify(result.value[0]) === lookupKeyStr) {
        lookupValue = result.value[1];
      }
    }

    return lookupValue;
  }
});

const exampleMap = new Map([[{a: 1}, 2]]);

console.log(exampleMap.deepCheck({a: 1}));

Very late to the party, but this library does exactly what you're looking for: https://github./adamhamlin/deep-equality-data-structures

const map = new DeepMap([[{a: 1}, 2]]);
console.log(map.get({a: 1})); // Prints: 2

Full disclosure: I am the library author

You need to somehow pare against or get a reference to the exact object that was used as the Map key. One option would be to iterate over the Map's entries, and check for one whose key has an a value of 1:

const exampleMap = new Map([[{a: 1}, 2]]);
const foundEntry = [...exampleMap.entries()]
  .find(([key, val]) => key.a === 1);
  
console.log(
  foundEntry
  ? foundEntry[1]
  : 'Not found!'
);

Another approach:

const exampleMap = new Map([[{a: 1}, 2]]);
const foundKey = [...exampleMap.keys()]
  .find(key => key.a === 1);
  
console.log(
  foundKey
  ? exampleMap.get(foundKey)
  : 'Not found!'
);

与本文相关的文章

发布评论

评论列表(0)

  1. 暂无评论