在循环中执行异步XMLHttpRequests,并在所有请求完成后继续

时间:2017-12-26 00:30:48

标签: javascript xmlhttprequest

我正在尝试在for loop中执行异步XMLHttpRequests,它会加载一堆JSON。

要确保在调用nextFunction()之前所有XMLHttpRequests请求都已完成,我正在使用xhr.onreadystatechange。 这似乎工作正常,但我的nextFunction()正在最后一次XMLHttpRequest完成之前执行。

为什么会这样?我做错了什么?

var arrCurrency = ['Bitcoin', 'Ethereum', 'Litecoin', 'Ripple'];
arrJSON = [];
createCurrencyArray();

function createCurrencyArray() {

  var count = 0;

  for (var i = 0; i < arrCurrency.length; i++) {

    var url = 'https://api.coinmarketcap.com/v1/ticker/' + arrCurrency[i] + '/?convert=EUR';

    getJSON(url, parseJSON);

    function parseJSON() {
      var a = this.responseText;
      var b = a.slice(1, -1);
      var c = JSON.parse(b);
      arrJSON.push(c);
    }
  }

  function xhrSuccess() {
    this.callback.apply(this, this.arguments);
  }

  function getJSON(url, callback) {

    var xhr = new XMLHttpRequest();
    xhr.callback = callback;
    xhr.onload = xhrSuccess;
    xhr.onreadystatechange = function() {

      if (xhr.readyState == 4) {

        count++;
        console.log('state:' + xhr.readyState + ' JSON-' + count + ' loaded')

        if (count == arrCurrency.length) {
          console.log("Done: All JSONs loaded");
          nextFunction();
        }
      }
    }
    xhr.open("GET", url, true);
    xhr.send(null);
  }
}

function nextFunction() {
  console.log(arrJSON);
  console.log('array length: ' + arrJSON.length);
  console.log('array key 0: ' + arrJSON[0]["id"]);
  console.log('array key 1: ' + arrJSON[1]["id"]);
  console.log('array key 2: ' + arrJSON[2]["id"]);
  console.log('array key 3: ' + arrJSON[3]["id"]);
}

1 个答案:

答案 0 :(得分:0)

您应该在此使用promises并在所有请求完成后处理结果。

&#13;
&#13;
var arrCurrency = ['Bitcoin', 'Ethereum', 'Litecoin', 'Ripple'], arrJSON = [], all = [];

function nextFunction() {  
  console.log('array length: ' + arrJSON.length);
  arrJSON.forEach((entry,idx)=>{console.log('array-key '+idx+': ' + arrJSON[idx].id); });
  console.log(arrJSON);
}

function parseJSON(jsn) {
    var obj = jsn.slice(1, -1);
    var jsnObj = JSON.parse(obj);
    arrJSON.push(jsnObj);
}
   
for(i=0; i < arrCurrency.length; i++){    
  
    var p = new Promise(function(resolve,reject){  
    var oReq = new XMLHttpRequest();	
	  oReq.addEventListener("load", function(xhr){ resolve(xhr);});
	  oReq.addEventListener("error",  function(b){reject(this);});
	  oReq.open('GET', ['https://api.coinmarketcap.com/v1/ticker/' + arrCurrency[i] + '/?convert=EUR'].join(""), true);
		oReq.send();   
   });
    all.push(p);
} // loop end
  
Promise.all(all).then(values => {
  values.forEach(function(p){parseJSON(p.currentTarget.responseText);});
  nextFunction();
});
&#13;
&#13;
&#13;