performance - Set of pairs of numbers in Javascript - Stack Overflow

ES6 has a new Set data structure for storing sets of unique objects. However it is based on object refe

ES6 has a new Set data structure for storing sets of unique objects. However it is based on object references as opposed to value parisons. As far as I can tell this makes it impossible to have a set of pairs of numbers without stringifying.

For example, typing in Chrome's console (needs Chrome 38+):

> var s = new Set();
< undefined
> s.add([2, 3]);
< Set {[2, 3]}
> s.has([2, 3])
< false          <--- was hoping for 'true'

This appears to be by design: since I passed a different array of [2, 3] to has(), it returns false, because although the contents is the same it only looks at object references, and I allocated a new and different array to pass to has(). I would need to store a reference to the original array I passed to add() to check with has(), but this is not always possible. For example if the number pairs represent co-ordinates, I might need to check if the set has [obj.x, obj.y], but this will always return false since it allocates a new array.

The workaround is to stringify the arrays and key on strings like "2, 3" instead. However in something performance-sensitive like a game engine, it is unfortunate if every set access needs to make a string allocation and convert and concatenate number strings.

Does ES6 provide any feature to solve this problem without stringifying, or is there any feature on the horizon with ES7 that could help as well?

ES6 has a new Set data structure for storing sets of unique objects. However it is based on object references as opposed to value parisons. As far as I can tell this makes it impossible to have a set of pairs of numbers without stringifying.

For example, typing in Chrome's console (needs Chrome 38+):

> var s = new Set();
< undefined
> s.add([2, 3]);
< Set {[2, 3]}
> s.has([2, 3])
< false          <--- was hoping for 'true'

This appears to be by design: since I passed a different array of [2, 3] to has(), it returns false, because although the contents is the same it only looks at object references, and I allocated a new and different array to pass to has(). I would need to store a reference to the original array I passed to add() to check with has(), but this is not always possible. For example if the number pairs represent co-ordinates, I might need to check if the set has [obj.x, obj.y], but this will always return false since it allocates a new array.

The workaround is to stringify the arrays and key on strings like "2, 3" instead. However in something performance-sensitive like a game engine, it is unfortunate if every set access needs to make a string allocation and convert and concatenate number strings.

Does ES6 provide any feature to solve this problem without stringifying, or is there any feature on the horizon with ES7 that could help as well?

Share Improve this question asked Sep 21, 2014 at 13:06 AshleysBrainAshleysBrain 22.7k16 gold badges91 silver badges125 bronze badges
Add a ment  | 

3 Answers 3

Reset to default 5

It is not perfectly optimal for very pute-intensive tasks, but you could use a concatenated string using template literals for a more idiomatic approach that still maintains efficiency, e.g.

set.add(`${x}_${y}`);

and retrieval:

set.get(`${i}_${j}`);

(note I've purposely avoided use of , as a delimeter since it can be confusing in some fields such as finance).

Another thing that could be done is grabbing the width of the first dimension to flatten an array if you know the bounds e.g.

set.get(x+y*width);

or if you're working with small numbers in general (not exceeding 10,000s) and don't know what the max width would be, you could use an arbitrary very large number. This is slightly less optimal but still better than string concat:

set.get(x+y*Math.floor(Math.sqrt(Number.MAX_SAFE_INTEGER)));

Again, these are not perfect solutions since they do not work with very large numbers where x*y may exceed Number.MAX_SAFE_INTEGER, but they are some things in your toolbox without needing to know a fixed array size.

[Super late here, but since ES7 had not fixed things after all and I noticed this was not specifically mentioned if others are weighing the pros/cons, two approaches (the first explicitly does not solve, the second may possibly)]

As you've noted [2, 3] === [2, 3] is false, meaning you can't use Set like this; however, is Set really the best option for you?

You may find that using a two-level data structure like this will be better for you

var o = {};

function add(o, x, y) {
    if (!o[x]) o[x] = {};
    o[x][y] = true;
}

function has(o, x, y) {
    return !!(o[x] && o[x][y]);
}

function del(o, x, y) {
    if (!o[x]) return;
    delete o[x][y];
    // maybe delete `o[x]` if keys.length === 0
}

You could do a similar structure with a Map pointing to Sets if you wanted to use ES6

is there any feature on the horizon with ES7 that could help as well?

There is a proposal in ECMAScript 7 to add Value Objects. Basically, it's a new immutable data type where identical value objects are pared by value, not by reference.

Depending on what kinds of value objects are implemented and/or if custom ones can be defined, they may solve this issue.

发布者:admin,转转请注明出处:http://www.yc00.com/questions/1743661395a4486257.html

相关推荐

  • performance - Set of pairs of numbers in Javascript - Stack Overflow

    ES6 has a new Set data structure for storing sets of unique objects. However it is based on object refe

    1天前
    10

发表回复

评论列表(0条)

  • 暂无评论

联系我们

400-800-8888

在线咨询: QQ交谈

邮件:admin@example.com

工作时间:周一至周五,9:30-18:30,节假日休息

关注微信