-
Notifications
You must be signed in to change notification settings - Fork 27.4k
Decouple $q in AngularJS 2.0 (and/or switch to bluebird) #6697
Comments
We're probably going to be using ES6 promises in Angular 2.0. I can't promise that the digest cycle will be separated from $q in 1.x, but that's really the big thing that gets in the way of allowing a pluggable interface, as far as I'm aware. |
+1 a pluggable promise interface would be great Just a comment on native promises, They are in ES six however they don't have the superb long stacktrace support that bluebird has. Also, bluebird can do 1 million promises per second while native promises and chrome are much slower at 25,000. That might not matter to most people but it may matter to somebody. |
Pluggable would be nice, but for the moment, it's not super feasible, because promises are tied to digest cycle. We can do this better for 2.0, but it might be very hard to remove this glue from 1.x, and that is the major barrier. In addition, it's very easy to mock the promise implementation in angular so that it can be used in tests. With a pluggable system, this is greatly complicated, and these things hurt. In any case, we can probably use Zone.js in 1.x to enable longer stack traces like Bluebird, even with the native $q implementation. I haven't yet looked at how difficult this would be, but it seems like something which is backportable to 1.x. It's true that we could probably use a better nextTick implementation which would perform better in the browser (like setImmediate, when available), among other things, but the real performance killer of the promise implementation in Angular isn't waiting for the next tick, but performing an app-wide digest. So even with Bluebird, if it's tied to the digest cycle, you wouldn't really see an improvement here for 1,000,000 promises simultaneously. |
@caitp thanks for the excellent info and I didn't know about Zone.js. |
+1 for decoupling it. I'd love to use a stronger promise implementation. |
@llambda "Also, bluebird can do 1 million promises per second while native promises and chrome are much slower at 25,000." That's crazy! I'd expect native browser methods to be an order of magnitude faster. Does that data come from Bluebird's page or somewhere else? |
It comes from the fact the guy writing Bluebird has very intimate knowledge of JS engines :) Also, JS isn't slower than native when you understand the JIT in these cases. |
V8's promises are implemented (mostly) in JS, anyways, so it's true that there is very little difference in these cases. |
To clarify, I'm against including Bluebird in Angular, I'm for decoupling the architecture here, and allowing users to easily hook their own implementation. |
Built-ins need to adhere to ridiculous semantic complexity which only gets worse as more features get added into the language. The spec is ruthless in that it doesn't leave any case as "undefined behavior" - what happens when you use splice on an array that has an indexed getter that calls Object.observe on the array while the splice is looping? If you implemented your own splice, then you probably wouldn't even think of supporting holed arrays, observable arrays, arrays with funky setters/getters and so on. Your splice would not behave well in these cases but that's ok because you can just document that. Additionally, since you pretty much never need the return value of splice, you can just not return anything instead of allocating a wasted array every time (you could also make this controllable from a parameter if needed). Already with the above, you could probably reach the perf of "native splice" without even considering the fact that user code is actually optimized and compiled into native code. And when you consider that, you are way past any "native" except when it comes to special snowflakes like Thirdly, built-ins are not magic that avoid doing any work, they are normal code implemented by humans. This is biggest reason for the perf difference specifically in the promise case - bluebird is extremely carefully optimized and tuned to V8 optimizing compiler expectations whereas the V8 promise implementation is looking like it's directly translated from the spec pseudo-code, as in there is no optimization effort at all. @jdalton has been trying to debunk the "natives are fast" myth for years and might have additional points. |
👍 for allowing the devs choice. To address @petkaantonov point, he's right emerging native API's rarely have a focus on performance as implementers are more concerned with simply getting the API working correctly. Native API is no guarantee for performance. Making those calls requires profiling and weighing the pros/cons of native vs. custom implementations. |
+1 for pluggable promises! |
+1 |
1 similar comment
+1 |
👍 👍 👍 👍 👍 👍 👍 👍 👍 👍 |
+1 |
1 similar comment
+1 |
Angular 2.0 has its own repository and is not going to use $q nor reimplement their own version of Promise/A+. I think this can be closed now |
BTW it turns out to be quite easy to use Bluebird with Angular 1.x; Bluebird has a setScheduler hook that supports exactly this (example via StackOverflow). I just started using this and am impressed at how easy it was to drop in and how big the benefits are. |
@metamatt actually more like http://plnkr.co/edit/ggMzDwnkzMOc0MccdVEu?p=preview in order to really decouple $q (this does the same as your example - only it replaces all the $q promises internally too). |
@benjamingr oh yes even better. Thanks for educating everyone on this. |
Hmm. I tried (a riff on) that and found it does indeed work, but there are a bunch of places both in my own code (mostly related to |
+1 |
1 similar comment
+1 |
+1 |
#12533 (comment) provides a way to replace |
AngularJS currently uses a shim of q. I haven't dug into the actual implementation, but I think it might be interesting to review a possible change for the 2.0 milestone.
I think AngularJS should use an agnostic
$Promise
service accross the framework, and extract the current$q
shim out of the core. Letting user choose their Promise framework. Thanks to the unified promise spec, this should be doable.Also, I would strongly advise to pick bluebird over q for the future.
I've been doing NodeJS work for a couple of years, and switching from q to the very well-crafted bluebird has been a really pleasant experience. The API is far superior and the performance is unmatched.
Regarding performance, have a look at this article.
The text was updated successfully, but these errors were encountered: