JSON数组未定义&诺巴蓝鸟

时间:2016-08-14 22:06:23

标签: json node.js promise bluebird

我正在使用Promise bluebird来处理来自文件的json数组对象。如果我想将数据存储在json数组(称为list)中并在最终进程中返回,则会出现问题。 列表返回后甚至在最终过程中,列表为空/未定义。运行代码,我总是有一个非假的值,它会触发列表中json的添加/推送。

你能帮我解决这个问题吗?您将在下面找到我的代码。

提前致谢!!!

var Promise = require('bluebird');
var join = Promise.join;
var fs = Promise.promisifyAll(require("fs"));

fs.readdirAsync(dir).map(function (filename) {
    return fs.readFileAsync(dir + "/" + filename, "utf8");
}).then(function(result){
    var list=[];

    result.map(function(row, index){
      Promise.coroutine(function*() {
        update(row, index).then(function(value){
          if (value!=false){
            var trade_update = new updated_Item(row.ID, row.Quantity, row.Price, row.Remark);
            list.push(trade_update);
            console.log(JSON.stringify(list));          <-- This works. It gives me data
          }
          return list;
        })
      })();
    });
    console.log('list: ' + JSON.stringify(list));        <-- output:    list:[]
    return list;
}).finally(function(result){
  console.log('Final outcome: '+ ' ' + JSON.stringify(result));      <-- output: Final outcome: undefined
})

在Samuel的帮助下,我的代码现在是:

var Promise = require('bluebird');
var join = Promise.join;
var fs = Promise.promisifyAll(require("fs"));


function updateOrder(done){
  fs.readdirAsync(dir).map(function (filename) {
      return fs.readFileAsync(dir + "/" + filename, "utf8");
  }).then(function(result){
    var list=[];

    result.map(function(row, index){
      Promise.coroutine(function*() {
        update(row, index).then(function(value){
          if (value!=false){
            var trade_update = new updated_Item(row.ID, row.Quantity, row.Price, row.Remark);
            list.push(trade_update);
            done(list);      
          }
        })
      })();
    });

    //done(list);   <--if I put the done callback here, it will give me an empty list. I though once the result.map finished processing all the values give me the end result.
  }
}

updateOrder(function(resultList){
  console.log('List' + JSON.stringify(resultList));
})

每次列表更新(推送)时,此代码都会给我整个resultList。

一旦函数updateOrder完成,我会在结尾处收到resultList。

1 个答案:

答案 0 :(得分:2)

如评论中所述。 Promise.coroutine是异步的,这意味着在代码到达之后,结果不会直接返回。这几乎解释了您所看到的现象,即您在代码中得到的后一个打印语句表明list未定义。

你可以做的是将你在那里得到的整个代码包装在一个函数中,然后添加一个callback函数作为async函数的参数,以便在它完成任务时调用,同时返回填充后的列表返回以供以后处理。

我为你的情况编写了一个伪代码,遗憾的是我无法在我的IDE上测试它,但是概念就在那里,它应该可以工作。

考虑我的伪代码:

var Promise = require('bluebird');
var join = Promise.join;
var fs = Promise.promisifyAll(require("fs"));

// Wrap everything you got into a function with a `done` parameter (callback fn)
function doStuff(done) {
  fs.readdirAsync(dir).map(function (filename) {
    return fs.readFileAsync(dir + "/" + filename, "utf8");
  }).then(function(result){
    var list=[];

    result.map(function(row, index){
      Promise.coroutine(function*() {
        update(row, index).then(function(value){
          if (value!=false){
            var trade_update = new updated_Item(row.ID, row.Quantity, row.Price, row.Remark);
            list.push(trade_update);
          }
          done(list);
        })
      })();
    });
  }).finally(function(result){
    console.log('File read finish, but this doesnt mean I have finished doing everything!');
  })
}

// call your function and provide a callback function for the async method to call 
doStuff(function(resultList) {
  console.log('list: ' + JSON.stringify(resultList));
  // Continue processing the list data.
});