Q承诺以正确的顺序进行链接

时间:2016-04-22 13:21:15

标签: javascript node.js promise q

我有一个异步函数需要以正确的顺序多次调用。这是关于将图像上传到服务器,但就像我说的那样,图像应该以正确的顺序上传。

我的功能如下:

  function uploadImage(sku,fileName) {
    console.log("Uploading image for sku="+sku+", imageName="+fileName);

    var deferred = Q.defer();

    var readStream = fs.createReadStream(rootPath+'/data/'+sku+"/"+fileName);
    var req = request.post('http://localhost:3000/'+sku+'/upload/'+fileName);

    readStream.pipe(req);

    req.on('end', function() {
      console.log("Image imageName="+fileName+" uploaded successfully.");
      db.updateLastUploadedImage(sku,fileName).then(function (res) {
        if(res) {
          console.log("Table watches for sku="+sku+" updated.");
          deferred.resolve(sku);
        }
      });
    });

    req.on('error',function(err) {
      deferred.reject(err);
    });

    return deferred.promise;
  }

我试图通过链接https://github.com/kriskowal/q中记录的承诺来启动它,但它运行不正常。不知怎的,我没有来到“那时”的街区。

所以我尝试创建一个递归函数,但它也没有进入函数调用的“then”块。

只有这种方法有效,但它没有按正确的顺序运行。

function uploadImages(sku) {
    var promises = [];

    for(var x=0; x<10; x++) {
      promises.push(uploadImage(sku,(x+1)+".jpg")));
    }

    return Q.all(promises).then(function (res) {
      return sku;
    });
}

我的递归解决方案如下所示:

function uploadImages(sku,current,max) {
    var deferred = Q.defer();

    if(current<=max) {
      uploadImage(sku,current+'.jpg').then(function (res) {
        if(res) {
          uploadImages(sku,current+1,max);
        }
      }, function (err) {
        deferred.reject();
      });
    } else {
      deferred.resolve(sku);
    }

    return deferred.promise;
 }

我正在寻找的是这样的东西(但这不是实施的方式):

 return uploadImage(sku,"1.jpg").then(function(res) {
      return uploadImage(sku,"2.jpg").then(function(res) {
        return uploadImage(sku,"3.jpg").then(function(res) {
          return uploadImage(sku,"4.jpg").then(function(res) {
            return uploadImage(sku,"5.jpg").then(function(res) {
              return uploadImage(sku,"6.jpg").then(function(res) {
                return uploadImage(sku,"7.jpg").then(function(res) {
                  return uploadImage(sku,"8.jpg").then(function(res) {
                    return uploadImage(sku,"9.jpg").then(function(res) {
                      return uploadImage(sku,"10.jpg").then(function(res) {
                        return sku;
                      });
                    });
                  });
                });
              });
            });
          });
        });
      });
    });

那么出于我的目的,最佳做法是什么?

2 个答案:

答案 0 :(得分:2)

异步调用没有“正确顺序”的概念,因为它们只是 - 异步,它们可以在任何时候结束。

Q.all(promises).then(...)中的回调中,您应该按照所做的顺序获得响应,但是由于它们的异步,控制台日志的顺序可能不会处于相同的顺序性质。

在您的情况下,您可以递归地执行此操作:

function uploadFiles(sku, current, max) {
     return uploadImage(sku, current + '.jpg').then(function (sku) {
          if (current > max) { return sku; }

          return uploadFiles(sku, current + 1, max);
     });
}

// use it
uploadFiles('SOME_SKU', 1, 10).then(function (sku) {
    // ALL DONE!
});

答案 1 :(得分:-2)

尝试从承诺中捕获异常。

return Q.all(promises).then(function (res) {
  return sku;
})
.catch(function (error) {
    // Handle any error from all above steps
});