Обещай, что решит и отвергнет - PullRequest
0 голосов
/ 26 мая 2020

Кажется, что мое Обещание одновременно возвращает истину и ложь. Консоль возвращает «undefined», а затем прямо под ним возвращает «что-то пошло не так». Данные возвращаются под ними, показывая, что на самом деле он не ожидает выполнения обещания.

Вот вызываемая функция:

module.exports = (url) => {
  return new Promise((resolve, reject) => {
    axios({
      method: 'get',
      url: url
    })
      .then(response => {
        const html = response.data
        const $ = cheerio.load(html)
        const songtable = $('.chart-list__elements > li')
        const topsongs = []
        songtable.each(function () {
          const rank = $(this).find('.chart-element__rank__number').text()
          if (rank == 11) return false;
          const name = $(this).find('.chart-element__information__song').text()
          const artist = $(this).find('.chart-element__information__artist').text()

          topsongs.push({
            rank,
            name,
            artist
          })
        })
        resolve()
        return topsongs;
      })
      .catch(reject("something went wrong"))
    })
}

От вызывающего:

componentDidMount() {
    const top_songs = topsongs('https://www.billboard.com/charts/hot-100')
    .then(console.log(top_songs))
    .catch(err => console.log(err))
  }

Спасибо, я новичок в Promises и пробовал почти все способы сделать это. Причина того, что у меня есть Promise, несмотря на вызов asyn c ax ios (), заключается в том, что он не выполнялся асинхронно и возвращал неопределенные данные.

Ответы [ 2 ]

3 голосов
/ 26 мая 2020
.catch(reject("something went wrong"))

Вам необходимо передать функцию в catch.

Вы немедленно вызываете reject и передаете его возвращаемое значение.


Вы также используете антишаблон вложенного обещания.

axios возвращает обещание. Нет необходимости создавать еще один.


module.exports = (url) =>
  axios({
    method: "get",
    url: url,
  })
    .then((response) => {
      const html = response.data;
      const $ = cheerio.load(html);
      const songtable = $(".chart-list__elements > li");
      const topsongs = [];
      songtable.each(function () {
        const rank = $(this).find(".chart-element__rank__number").text();
        if (rank == 11) return false;
        const name = $(this).find(".chart-element__information__song").text();
        const artist = $(this)
          .find(".chart-element__information__artist")
          .text();
        topsongs.push({
          rank,
          name,
          artist,
        });
      });
      return topsongs;
    })
    .catch(() => {throw "something went wrong"});

(Замена выданной ошибки на generi c «что-то пошло не так» кажется бесполезной. Возможно, вам будет лучше без этого вызова catch вообще)

0 голосов
/ 26 мая 2020

У вас уже есть обещание, просто верните его.

  return axios({
      method: 'get',
      url: url
    })
      .then(response => {
        const html = response.data
        const $ = cheerio.load(html)
        const songtable = $('.chart-list__elements > li')
        const topsongs = []
        songtable.each(function () {
          const rank = $(this).find('.chart-element__rank__number').text()
          if (rank == 11) return false;
          const name = $(this).find('.chart-element__information__song').text()
          const artist = $(this).find('.chart-element__information__artist').text()

          topsongs.push({
            rank,
            name,
            artist
          })
        })
        return topsongs;
      })

И только для «syntacti c sugar» async/await упрощает чтение:

module.exports = async (url) => {
   const { data } = await axios({method:'get',url});
   const $ = cheerio.load(data);

   ...

   return topsongs;
}
...