Как структурировать вложенные Обещания

У меня есть ситуация, когда я думаю, что единственный выбор для меня - это вложить некоторые Обещания друг в друга. У меня есть Обещание, которое необходимо выполнить, и метод, который делает что-то, пока это Обещание не будет выполнено. Что-то вроде этого:

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);
Другие вопросы по тегам