使while循环同步

时间:2013-04-09 12:42:17

标签: javascript node.js asynchronous

我有一段代码

var page = 2;
var last_page = 100;
while(page <= last_page) {
  request("http://some_json_server.com/data?page=" + page, function (error, response, body) {
    if (!error && response.statusCode == 200) {
      store_data(body)
    }
    page++;
  });                
}

我已经完成了以下操作,但实际上并没有检索任何内容。我这样做了吗?

var page = 2;
var last_page = 100;
while(page <= last_page) {
var async_arr = [];
async_arr.push(
  function(next) {
    request("http://some_api_url?page=" + page, function (error, response, body) {
      if (!error && response.statusCode == 200) {
        store_data(body);
      }
    });
  }
);

async.series(
  async_arr, done
);

3 个答案:

答案 0 :(得分:53)

使用while,您将获得一个繁忙的循环,这在Node中是有目的的。

改为使其成为递归函数。每次通话都将在单独的时间点完成。

var page = 2;
var last_page = 100;

(function loop() {
    if (page <= last_page) {
        request("/data?page=" + page, function (error, response, body) {
            if (!error && response.statusCode == 200) {
                store_data(body)
            }
            page++;
            loop();
        });
    }
}());

答案 1 :(得分:23)

您正在寻找async.whilst()。这个解决方案假设您实际上想要在另一个之后执行每个请求。正如@UpTheCreek提到的那样(编辑:我引用的评论已被编辑),很可能会异步进行并使用async.parallel跟踪每个结果。

var page = 2,
    lastPage = 100;

async.whilst(function () {
  return page <= lastPage;
},
function (next) {
  request("http://some_json_server.com/data?page=" + page, function (error, response, body) {
    if (!error && response.statusCode == 200) {
      store_data(body)
    }
    page++;
    next();
  });
},
function (err) {
  // All things are done!
});

答案 2 :(得分:0)

您还可以将while循环包装在异步中,并在满足您的诺言解决/条件后中断...

    const request = require("request")

    ;(async()=>{

    let results = []
    while(true){
      await new Promise(resolve => {
        request('http://www.seanbehan.com/', (err, resp, body)=>{
          console.log(new Date, 'Downloading..')
          results.push(body)
          resolve(body)
        })
      })

      if(results.length >= 5){
        break
      }
    }

    console.log(results)
  })()