What is the difference between:
new Promise(function(res, rej) { res("aaa"); }) .then(function(result) { return "bbb"; }) .then(function(result) { console.log(result); });
and this:
new Promise(function(res, rej) { res("aaa"); }) .then(function(result) { return Promise.resolve("bbb"); }) .then(function(result) { console.log(result); });
I'm asking as I'm getting different behaviour Using Angular and $http service with chaining .then(). A bit too much code hence first the example above.
Promise.resolve()
in the second example is unnecessary.
then
handler, in fact, it's a key aspect of the promises spec that you can do that.
new Promise((res, rej) => { return fetch('//google.com').then(() => { return "haha"; }) }).then((result) => alert(result));
This code will just hang (not resolved forever). But if I change return "haha";
to return res("haha");
then it'll work and alert "haha". Didn't the fetch().then() already wrap "haha" into a resolved promise?
res
parameter of the function passed instead of returning the result. Returning anything inside of the function passed to the Promise constructor will be ignored
In simple terms, inside a then
handler function:
A) When x
is a value (number, string, etc):
return x is equivalent to return Promise.resolve(x) throw x is equivalent to return Promise.reject(x)
B) When x
is a Promise that is already settled (not pending anymore):
return x is equivalent to return Promise.resolve(x), if the Promise was already resolved. return x is equivalent to return Promise.reject(x), if the Promise was already rejected.
C) When x
is a Promise that is pending:
return x will return a pending Promise, and it will be evaluated on the subsequent then.
Read more on this topic on the Promise.prototype.then() docs.
The rule is, if the function that is in the then
handler returns a value, the promise resolves/rejects with that value, and if the function returns a promise, what happens is, the next then
clause will be the then
clause of the promise the function returned, so, in this case, the first example falls through the normal sequence of the thens
and prints out values as one might expect, in the second example, the promise object that gets returned when you do Promise.resolve("bbb")
's then is the then
that gets invoked when chaining(for all intents and purposes). The way it actually works is described below in more detail.
Quoting from the Promises/A+ spec:
The promise resolution procedure is an abstract operation taking as input a promise and a value, which we denote as [[Resolve]](promise, x). If x is a thenable, it attempts to make promise adopt the state of x, under the assumption that x behaves at least somewhat like a promise. Otherwise, it fulfills promise with the value x. This treatment of thenables allows promise implementations to interoperate, as long as they expose a Promises/A+-compliant then method. It also allows Promises/A+ implementations to “assimilate” nonconformant implementations with reasonable then methods.
The key thing to notice here is this line:
if x is a promise, adopt its state [3.4] link: https://promisesaplus.com/#point-49
then
handler returns a promise. +1 for the spec reference.
[[Resolve]]
is called both on then
ables and values so essentially it wraps a value with the promise so return "aaa"
is the same as return Promise.resolve("aaa")
and return Promise.resolve("aaa")
is the same as return Promise.resolve(Promise.resolve("aaa"))
- since resolve is idempotent calling it on a value more than once has the same result.
"aaa"
and return Promise.resolve("aaa")
are interchangeable in then
ables in any cases?
Both of your examples should behave pretty much the same.
A value returned inside a then()
handler becomes the resolution value of the promise returned from that then()
. If the value returned inside the .then
is a promise, the promise returned by then()
will "adopt the state" of that promise and resolve/reject just as the returned promise does.
In your first example, you return "bbb"
in the first then()
handler, so "bbb"
is passed into the next then()
handler.
In your second example, you return a promise that is immediately resolved with the value "bbb"
, so "bbb"
is passed into the next then()
handler. (The Promise.resolve()
here is extraneous).
The outcome is the same.
If you can show us an example that actually exhibits different behavior, we can tell you why that is happening.
Promise.resolve();
vs return;
?
undefined
instead of "bbb"
.
You already got a good formal answer. I figured I should add a short one.
The following things are identical with Promises/A+ promises:
Calling Promise.resolve (In your Angular case that's $q.when)
Calling the promise constructor and resolving in its resolver. In your case that's new $q.
Returning a value from a then callback.
Calling Promise.all on an array with a value and then extract that value.
So the following are all identical for a promise or plain value X:
Promise.resolve(x);
new Promise(function(resolve, reject){ resolve(x); });
Promise.resolve().then(function(){ return x; });
Promise.all([x]).then(function(arr){ return arr[0]; });
And it's no surprise, the promises specification is based on the Promise Resolution Procedure which enables easy interoperation between libraries (like $q and native promises) and makes your life overall easier. Whenever a promise resolution might occur a resolution occurs creating overall consistency.
Promise.resolve().then(function(){ return x; });
? I found a snipped doing something similar (it called a function inside the then
block). I thought it was more or less like doing a timeout, but it's a bit faster. jsben.ch/HIfDo
with
block over an object or proxy with an x
property accessor that throws an exception. In that case Promise.resolve(x) would cause a thrown error but Promise.resolve().then(function(){ return x; });
would be a rejected promise since the error is thrown in a then
).
if (validator) { Promise.resolve().then(() => { this._cdRef.markForCheck(); }); }
. Here the promise isn't assigned, so what's the point? A timeout would have (more or less) the same effect, or not?
The only difference is that you're creating an unnecessary promise when you do return Promise.resolve("bbb")
. Returning a promise from an onFulfilled()
handler kicks off promise resolution. That's how promise chaining works.
Just a point. The 'resolve' function does not return as you can see in a debugger. It seems to return if it is the last statement. 'Resolve' will set the promise as fullfilled but continue execution if there are more statements below.
As you can read here: 'Why does javascript ES6 Promises continue execution after a resolve?'
This was very confusing for me because it is not explained in most examples. So from now on, I have to remember to use 'return resolve(value)' or an 'if resolve(value) else ...other code' or just use 'resolve' as the last statement.
Success story sharing
return
changes depending on the settled value of the Promise. When writing the code you don't really know if it will resolve or reject. Of course, you can have a try-catch to handle rejects and then re-throw, but that's a different scenario.