-
Notifications
You must be signed in to change notification settings - Fork 30k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
lib: use Object.create(null) directly #11930
Conversation
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.
Code LGTM. I worry that this will make debugging more difficult though.
There seems to be some regressions in the event benchmarks... See numbersimprovement confidence p.value events/ee-add-remove.js n=250000 -43.29 % *** 4.520847e-30 events/ee-emit-multi-args.js n=2000000 1.57 % 2.325176e-01 events/ee-emit.js n=2000000 8.37 % ** 6.855389e-03 events/ee-listener-count-on-prototype.js n=50000000 -96.06 % *** 8.635026e-26 events/ee-listeners-many.js n=5000000 -17.51 % *** 3.849835e-27 events/ee-listeners.js n=5000000 -56.36 % *** 6.473337e-30 events/ee-once.js n=20000000 -51.46 % *** 4.847882e-33 |
Interesting. The V8 optimization is mainly designed to accommodate map-like usage of objects, by automatically using so-called "slow" properties based on a hashmap instead of trying to find hidden classes. This makes it work especially well with cases like HTTP headers, FS cache, and query string parsing that don't have a fixed pattern of object properties. On the other hand, an instance of Take the benchmark with the worst regression,
I believe having two or more events is a considerably more common case than just having one event, and therefore I think this change is still justified. benchmark diffdiff --git a/benchmark/events/ee-listener-count-on-prototype.js b/benchmark/events/ee-listener-count-on-prototype.js
index dfe7e27..1ef0f09 100644
--- a/benchmark/events/ee-listener-count-on-prototype.js
+++ b/benchmark/events/ee-listener-count-on-prototype.js
@@ -9,12 +9,12 @@ function main(conf) {
var ee = new EventEmitter();
- for (var k = 0; k < 10; k += 1)
- ee.on('dummy', function() {});
+ for (var k = 0; k < 2 * 10; k += 1)
+ ee.on(`dummy${k % 2}`, function() {});
bench.start();
for (var i = 0; i < n; i += 1) {
- ee.listenerCount('dummy');
+ ee.listenerCount(`dummy${i % 2}`);
}
bench.end(n);
} |
Can you elaborate? I doubt having a generic name like |
I meant specifically in heap snapshots. |
I think the benchmarks can add a few more configurations for different use cases. Note that the diff in #11930 (comment) also removes the loop invariant(not sure if V8 could perform LICM on that previously). Also +1 with @cjihrig, with |
What's up with the 'manyblankpairs' regression? |
@mscdex, it can be attributed to the initial cost of object creation:
|
Is there any perf difference with |
|
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.
changes LGTM but needs a rebase
The object is used as a structure, not as a map, which `StorageObject` was designed for.
After V8 5.6, using Object.create(null) directly is now faster than using a constructor. Refs: emberjs/ember.js#15001 Refs: https://crrev.com/532c16eca071df3ec8eed394dcebb932ef584ee6
6d1a545
to
9549e2d
Compare
@joyeecheung I added a commit that uses a dedicated class for |
Landed in d9b0e4c...cfc8422. |
PR-URL: nodejs#11930 Reviewed-By: Anna Henningsen <anna@addaleax.net> Reviewed-By: Colin Ihrig <cjihrig@gmail.com> Reviewed-By: Luigi Pinca <luigipinca@gmail.com> Reviewed-By: James M Snell <jasnell@gmail.com>
The object is used as a structure, not as a map, which `StorageObject` was designed for. PR-URL: nodejs#11930 Reviewed-By: Anna Henningsen <anna@addaleax.net> Reviewed-By: Colin Ihrig <cjihrig@gmail.com> Reviewed-By: Luigi Pinca <luigipinca@gmail.com> Reviewed-By: James M Snell <jasnell@gmail.com>
After V8 5.6, using Object.create(null) directly is now faster than using a constructor for map-like objects. PR-URL: nodejs#11930 Refs: emberjs/ember.js#15001 Refs: https://crrev.com/532c16eca071df3ec8eed394dcebb932ef584ee6 Reviewed-By: Anna Henningsen <anna@addaleax.net> Reviewed-By: Colin Ihrig <cjihrig@gmail.com> Reviewed-By: Luigi Pinca <luigipinca@gmail.com> Reviewed-By: James M Snell <jasnell@gmail.com>
@TimothyGu I'm seeing contradicting results on master with some more simple, direct benchmarks: 'use strict';
var n = 1e7;
var input = { mode: 0o666, flags: 'r', start: 5, end: 100 };
function StorageObject() {}
StorageObject.prototype = Object.create(null);
function copyObject(source) {
const target = new StorageObject();
for (var key in source)
target[key] = source[key];
return target;
}
function copyObject2(source) {
const target = Object.create(null);
for (var key in source)
target[key] = source[key];
return target;
}
var r;
console.time('copyObject');
for (var i = 0; i < n; ++i) {
r = copyObject(input);
r.mode >>>= 0;
}
console.timeEnd('copyObject');
console.time('copyObject2');
for (var i = 0; i < n; ++i) {
r = copyObject2(input);
r.mode >>>= 0;
}
console.timeEnd('copyObject2'); Results:
With just the first two properties in
With just the first property in
|
I should add that I just tried with V8 5.7 which was just merged into master and the results are similar. |
Ah, I just noticed that es/map-bench.js isn't actually benchmarking object creation, but instead property access. There is a much larger difference with object creation in I think we should revert the changes in this PR (except maybe the |
I also noticed the huge difference with object creation time but this also "fixes" something weird which happens when using instances of function StorageObject() {}
StorageObject.prototype = Object.create(null); I've described the behavior in #11868 (comment). @mscdex can you take a look? |
This is a breaking change. Maybe it's faster now, but it is not backwards compatible, e.g. it broke the following code:
now throws |
@twoi I imagine you upgraded from a release to another, can you give us the version numbers? Was this a patch/minor release or a major release? Thanks. |
We upgraded from 0.8.15 to 11.9 - finally, because we are using a native module that (pre-napi) needed to ported / rebuilt on every Node.js version upgrade, which include all the vast v8 refactorings. But this change apparently made it into Node.js version 8.0.0. It's filed under |
@twoi this PR is semver-patch. The removal of the prototype was done in #6092 which is marked as semver-major (v6). Is there anything actionable you ask for? I highly recommend to never use |
@BridgeAR Thanks for the advice. Agree. I don't have anything actionable, just wanted to make aware of the fact that this innocent looking change can (and in fact did) break someone's code. Which, it seems, people noticed during testing: https://github.com/nodejs/node/pull/6092/files#r58880619 I am adding more testing myself, so this won't catch me unawares anymore... Case closed. |
@BridgeAR I see. Makes sense. Thanks |
Since v8/v8@532c16e (included in v5.6), using
Object.create(null)
directly is now faster than using a constructor.Refs: emberjs/ember.js#15001
Refs: https://crrev.com/532c16eca071df3ec8eed394dcebb932ef584ee6
/cc @nodejs/v8
querystring
map-bench
Checklist
make -j4 test
(UNIX), orvcbuild test
(Windows) passesAffected core subsystem(s)
lib