Как структурировать вложенные Обещания
У меня есть ситуация, когда я думаю, что единственный выбор для меня - это вложить некоторые Обещания друг в друга. У меня есть Обещание, которое необходимо выполнить, и метод, который делает что-то, пока это Обещание не будет выполнено. Что-то вроде этого:
let promise = new Promise((resolve, reject) => {
// Do some stuff
});
doSomethingUntilPromiseisDone(promise);
Однако в моем обещании мне нужно выполнить другой метод, который возвращает другое обещание:
let promise = new Promise((resolve, reject) => {
fetchValue(url)
.then((value) => {
// Do something here
}).catch((err) => {
console.error(err);
});
});
doSomethingUntilPromiseisDone(promise);
Но сейчас, в fetchValue
метода then
Скажите, у меня есть другой метод, который мне нужен, чтобы выполнить это, угадайте, что, возвращает еще одно обещание:
let promise = new Promise((resolve, reject) => {
fetchValue(url)
.then((value) => {
saveToCache(value)
.then((success) => {
console.log('success!!');
resolve('success');
});
}).catch((err) => {
console.error(err);
});
});
doSomethingUntilPromiseisDone(promise);
Итак, в конце концов, у меня есть Обещание, в Обещании, в Обещании. Могу ли я как-то структурировать это так, чтобы оно было более простым? Кажется, что вложение их друг в друга противоречит предполагаемому подходу Promise к созданию цепочки.
2 ответа
Использование .then()
let doStuff = (resolve, reject) => {/* resolve() or reject() */};
let promise = new Promise(doStuff);
doSomethingUntilPromiseisDone(
promise
.then(value => fetchValue(url))
.then(value => value.blob())
.then(saveToCache)
)
.then(success => console.log("success!!"))
.catch(err => console.error(err))
Ты можешь использовать generator
чтобы сгладить ваши вложенные обещания ( Bluebird.couroutine или Generators)
//Bluebird.couroutine
const generator = Promise.coroutine(function*() {
try {
const value = yield fetchValue(url);
const success = yield saveToCache(value);
console.log('success:', success);
} catch(e) {
console.error(err);
}
}));
generator();
Каждая функция будет вызывать следующую с результатом метода ранее.
var promises = [1,2,3].map((guid)=>{
return (param)=> {
console.log("param", param);
var id = guid;
return new Promise(resolve => {
// resolve in a random amount of time
setTimeout(function () {
resolve(id);
}, (Math.random() * 1.5 | 0) * 1000);
});
}
}).reduce(function (acc, curr, index) {
return acc.then(function (res) {
return curr(res[index-1]).then(function (result) {
console.log("result", result);
res.push(result);
return res;
});
});
}, Promise.resolve([]));
promises.then(console.log);