How to chain multiple fetch() promises?

前端 未结 7 1424
佛祖请我去吃肉
佛祖请我去吃肉 2021-01-06 17:08

The following code fetches a json list and then does another fetch call for each list item to change their values. The problem is that it’s not done synchronously. “new” is

相关标签:
7条回答
  • 2021-01-06 17:27

    How to chain multiple fetch() promises?

    You do it like how you have been doing it, just append another .then()

    fetch(API_URL_DIARY)
    .then(response => response.json())
    .then(data => {
      console.log("old", data);
      return data;
    })
    .then(data => {
      data.forEach(function(e, index,array) {
        fetch(API_URL_FOOD_DETAILS + e.foodid)
        .then(response => response.json())
        .then(data => {
          array[index] = {...e, ...data};
          console.log("update");
        })
        .then(()=>{
          console.log("new", data)  
        })
      });
    });
    
    0 讨论(0)
  • 2021-01-06 17:27

    You will need a recursive function to do this.

        fetch(API_URL_DIARY)
        .then(response => response.json())
        .then(data => {
          console.log("old", data);
          return data;
        })
        .then(data => {
    
        recursiveFetch(data)
    
        });
    
    function recursiveFetch(initialData){
            e = initialData[initialData.length-1]; //taking the last item in array
            fetch(API_URL_FOOD_DETAILS + e.foodid)
            .then(response => response.json())
            .then(data => {
              array[index] = {...e, ...data};
              console.log("update");
              initialData.pop() // removing last item from array, which is already processed
              if(initialData.length > 0)
                 recursiveFetch(initialData)
            })
    }
    

    Note: This is an untested code.

    0 讨论(0)
  • 2021-01-06 17:28

    You shouldn't be using forEach here. The best solution is to use Promise.all which waits for an array of promises (fetch is a promise) to all resolve, after which you can process the data.

    Here I've created a dummy fetch function with some sample data to quickly show you how that works.

    const dummyObj = {
      main: [ { id: 1 }, { id: 2 }, { id: 5 } ],
    	other: {
        1: 'data1',
        2: 'data2',
        3: 'data3',
        4: 'data4',
        5: 'data5',
        6: 'data6',
        7: 'data7',
      }  
    }
    
    // The summy function simply returns a subset of the sample
    // data depending on the type and id params after 2 seconds
    // to mimic an API call
    function dummyFetch(type, id) {
      return new Promise((resolve) => {
        setTimeout(() => {
          resolve(id ? dummyObj[type][id] : dummyObj[type]);
        }, 2000);
      });
    }
    
    // In the first fetch we display the data, just
    // like you did in your example
    dummyFetch('main')
    .then(data => {
      console.log("old", data);
      return data;
    })
    .then(data => {
    
      // Instead of a forEach use Array.map to iterate over the
      // data and create a new fetch for each
      const promises = data.map(o => dummyFetch('other', o.id));
    
      // You can then wait for all promises to be resolved
      Promise.all(promises).then((data) => {
    
        // Here you would iterate over the returned group data
        // (as in your example)
        // I'm just logging the new data as a string
        console.log(JSON.stringify(data));
    
        // And, finally, there's the new log at the end
        console.log("new", data)
      });
    });

    Here's the async/await version:

    const dummyObj = {
      main: [ { id: 1 }, { id: 2 }, { id: 5 } ],
    	other: {
        1: 'data1',
        2: 'data2',
        3: 'data3',
        4: 'data4',
        5: 'data5',
        6: 'data6',
        7: 'data7',
      }  
    }
    
    function dummyFetch(type, id) {
      return new Promise((resolve) => {
        setTimeout(() => {
          resolve(id ? dummyObj[type][id] : dummyObj[type]);
        }, 2000);
      });
    }
    
    (async () => {
      const oldData = await dummyFetch('main');
      console.log("old", oldData);
      const promises = oldData.map(o => dummyFetch('other', o.id));
      const newData = await Promise.all(promises);
      console.log(JSON.stringify(newData));
      console.log('new', newData);
    })();

    0 讨论(0)
  • 2021-01-06 17:29

    The best way to go about this is to use Promise.all() and map().

    What map will do in this context return all the promises from fetch.

    Then what will happen is await will make your code execution synchronous as it'll wait for all of the promise to be resolved before continuing to execute.

    The problem with using forEach here is that it doesn't wait for asynchronous request to be completed before it moves onto the next item.

    The code that you should be using here is:

    fetch(API_URL_DIARY)
        .then(response => response.json())
        .then(data => {
            console.log("old", data);
            return data;
        })
        .then(async data => {
            await Promise.all(data.map((e, index, array) => {
                return fetch(API_URL_FOOD_DETAILS + e.foodid)
                    .then(response => response.json())
                    .then(data => {
                        array[index] = {...e, ...data};
                        console.log("update");
                    })
            }));
    
            console.log("new", data)
        });
    
    0 讨论(0)
  • 2021-01-06 17:31

    Storing multiple responses in a single array
    The following code fetches multiple keywords in queries and stores all the response of all three responses to the all array

    let queries = ["food", "movies", "news"]
    let all = []
    
    queries.forEach((keyword)=>{
      let [subres] = await Promise.all([fetch(`https://reddit.com/r/${keyword}/hot.json?limit=100`).then((response) => response.json())]);
      all.push(subres)
    })
    
    //now you can use the data globally or use the data to fetch more data
    console.log(all)
    
    0 讨论(0)
  • 2021-01-06 17:36

    fetch is a Promise. This is asyncronous call, so the "new" console.log runs before finished all the promises. Use Promise.all() for that.

    You can do this so:

    fetch(API_URL_DIARY)
      .then(response => response.json())
      .then(data => {
        console.log("old", data);
        return data;
      })
      .then(data => {
        return Promise.all(data.map(food =>
          fetch(API_URL_FOOD_DETAILS + food.foodid)
            .then(resp => resp.json())
            .then(json => {
              // do some work with json
              return json
            })
        ))
      })
      .then(data => console.log('new', data))
    
    0 讨论(0)
提交回复
热议问题