When comparing this benchmark with chrome 16 vs opera 11.6 we find that
- in chrome native bind is almost 5 times slower then an emulated version of bind
- in opera native bind is almost 4 times faster then an emulated version of bind
Where an emulated version of bind in this case is
var emulatebind = function (f, context) {
return function () {
f.apply(context, arguments);
};
};
Are there good reasons why there is such a difference or is this just a matter of v8 not optimizing enough?
Note: that emulatebind only implements a subset but that isn’t really relevant. If you have a fully featured and optimised emulated bind the performance difference in the benchmark still exists.
Based on http://jsperf.com/bind-vs-emulate/6, which adds the es5-shim version for comparison, it looks like the culprit is the extra branch and
instanceofthat the bound version has to perform to test if it’s being called as a constructor.Each time the bound version is run, the code that gets executed is essentially:
In the V8 source code, this check appears (inside
boundFunction) as(Plaintext link to v8natives.js for when Google Code Search dies.)
It is a bit puzzling that, for Chrome 16 at least, the es5-shim version is still faster than the native version. And that other browsers have rather varying results for es5-shim vs. native. Speculation: maybe
%_IsConstructCall()is even slower thanthis instanceof bound, perhaps due to crossing native/JS code boundaries. And perhaps other browsers have a much faster way of checking for a[[Construct]]call.