Q 承诺链接以正确的顺序做事

Q promises chaining to do things in correct order

本文关键字:顺序 承诺 链接      更新时间:2023-09-26

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

我的函数如下所示:

  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;
                      });
                    });
                  });
                });
              });
            });
          });
        });
      });
    });

那么,对于我的目的来说,最佳实践是什么?

异步调用没有"正确顺序"的概念,因为它们就是异步的,它们可以随时结束。

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!
});

尝试从承诺中捕获异常。

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