-
Notifications
You must be signed in to change notification settings - Fork 27.4k
perf(copy): use ES6 Map to track source/destination objects #13209
Conversation
2125ea0
to
4069853
Compare
|
||
// Object keys, must use instance as key and not call toString | ||
&& m.set(o, 2) === m && m.get(o) === 2 | ||
&& m.get({}) === undefined; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Are so specific tests necessary? Do you have examples of browsers that did it wrong?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Might not be necessary. I was more concerned about bad non-native shims / helpers. An is-native check might also work but I don't think that is done anywhere else...
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@mzgol The set
method in IE11 is not spec compliant as I found out when writing my own polyfills
https://connect.microsoft.com/IE/Feedback/Details/2003432
Edit Doesn't look like the return value from set
is used, so it probably doesn't matter here.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I guess I was trying to make sure get/set were fully compliant so ES6Map
could be used as if it was a real map. But it's already only a subset of ES6 (just get/set) so maybe those 2 methods don't have to be perfect either...
AFAICT, very few people rely (atm) on native Map implementations being available in the browser. I believe most people, either don't care or use a shim. Thus, I wouldn't make a change that makes But, I would be all for a change that has no effect for the majority of cases (or just a tiny impact) and improves the case where a native Map implementation is available. There are 3 categories of "usecases":
In case (1), we will harness the perf improvement. GOOD In case (2), we shouldn't degrade performance. If we can achieve that via the Case (3), which potentially applies to a large number of people atm, is what worries me. I would investigate how is browser support and how is the performance with some of the popular polyfills. (*): WRT browser support for Map, according to kangax the situation is good, but probably not good enough. IE10, Safari 7 and mobile browsers doesn't seem to have adequate support. Also, note that polyfills still might do some "patchwork" on browsers with good (but not complete) support, so there might be an impact there as well. If all is GOOD, I would go for this change. If not, I would wait for native implementations to catch up a little more. Just my 2c... |
@jbedard: Does this mean the |
BTW, changing this to string comparisons would make the fix for |
c8768d1 has been merged which avoids the regex. Could still be changed if it would help with Could probably change the map shim to remember the last key looked up. That would make things like like get+put only do one |
Also note that memory usage seems to go down 15-30% when using the native map. |
4069853
to
3010cd6
Compare
I've added a commit which avoids the double I'm still not sure if this is worth having a shim like this though. Would be great if others can try it out with different apps or test data... |
} | ||
}, | ||
set: function(key, value) { | ||
var idx = this._idx(key); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
can we abuse of the fact that we know we are never going to override a value?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I am thinking something like
set: function(key, value) {
this._keys.push(key);
this._values.push(value);
}
get: function(key) {
var idx = this._keys.lastIndexOf(key);
if (idx != -1) return this._values[idx];
}
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We could for this copy
specific use case. But I was trying to keep this shim equivalent to the native ES6 version so it could be used anywhere...
@gkalpak Actually, I was going to suggest that we consider using an implementation closer to |
@@ -103,6 +103,8 @@ | |||
"getBlockNodes": false, | |||
"createMap": false, | |||
"VALIDITY_STATE_PROPERTY": true, | |||
"testES6Map": true, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Redundant
@dcherman, not sure what you mean 😁 I don't think property lookups (in shape-changing, unoptimizable objects) are preferrable to If we don't, then we could replace the (presumably) slower Are my assumptions wrong? Did I miss something? |
Today We could update the I could try this if there's interest... |
@jbedard Primitives are easy - the stringified version of a primitive is itself the key since primitives are always Never considered the frozen object use case when I implemented my own Map polyfill, good thought. |
Good point, and I always assumed the frozen object case is why |
Still, ignoring the
Considering that the objects which |
For large collections I think that |
I don't know that much about VM internals, but is |
I think looping over every entry in an array is much slower then looking up one property, doesn't matter what VM you're using. |
Unless properties are stored in an array-like structure 😛 |
@gkalpak I don't know about VMs other than V8, but unoptimized objects fallback to dictionary/hashmap mode. Optimized objects use hidden classes that can point as a specific place in memory where a property is stored. Here's one article on the subject if you're interested: |
All of that said, we're debating how to implement a data-structure that is only unsupported in IE9/10 amongst the supported browser set I believe :) Any changes to the internals should be non-breaking since we have a well defined interface, so we can choose not to let this discussion hold up this improvement. |
I totally agree that if this improves performance (most of the time) - which it seems it does - we should defnitely get it in. But I would like to sort the So, I do believe we should sort this issues out before merging this. Do we need |
I run through all uses of |
Tests pass with the native Map, but some fail unexpectedly with |
Nevermind, it was my faulty implementation of |
So If |
Yes, it seems that |
I assume performance with the native Map is about the same or maybe better then We can probably improve performance of the shim by using |
Voilà #15114. |
Jason just stop providing PRs! |
@lgalfaso here's the ES6 Map change I promised.
When running f7bf118 the larger objects are 10-20% faster, but the smaller objects are a bit slower (0-25%, but the times are so small I don't think this means much). On my box that meant the larger objects were ~400ms faster, the smaller were 1-3ms slower. Memory usage also goes down 15-30% (30 on the more complicated cases, 15 on the simpler ones).
When using the shim (manually enabling it on chrome) the larger objects are 20-25% slower, the smaller objects are a bit slower just like when using the native
Map
. This is because the shim causes an extraindexOf
call compared to before this change (one onget
, a second onset
). This extraindexOf
could be avoided if we remove the check if the key already exists in the shimset
method, but then this wouldn't be a properMap
shim (or add a magic 3rd param toset
called "iAlreadyKnowTheKeyDoesNotExist` and always pass true? :P). EDIT: 105dd5d removes this extra 20-25% when using the shimNote that while profiling
isTypedArray
is what stands out the most, notMap
orindexOf
, see #12054.