3

这类似于我今天发布的问题,但需要串行链接请求。我有两个异步请求,第二个请求需要第一个请求的结果来发送查询。

var Db.get = function(key){
    var deferred = $q.defer();

     //send async req
    var req = ....
    req.success = function(d){
        deferred.resolve(d)
    };
    req.failure = function(d){
        deferred.reject(d)
    }

    return deferred.promise;
}

var someFn = function(id){
    Db.get(id, "abc")
        .then(function (d) {
            console.log("At 1")
            Db.get(d.id, "def")
                .then(function (d) {
                    console.log("At 2")
                    return d
                }, function (e) {
                    //error
                });
        }, function (e) {
            //error
        });

    console.log("At 3")
};

我应该想错了,因为我希望console.log("At 3")在返回之后永远不会在成功的情况下打印console.log("At 2")。但是当我运行时,在控制台中我看到了这些顺序

console.log("At 1")
console.log("At 3")
console.log("At 2")

我在想then会阻塞直到它得到承诺的响应(由 get() 返回)。因此, someFn 中的所有内容都是串行执行的。这个假设是错误的吗?链接两个使用承诺的异步操作以串行运行的最佳方法是什么。

谢谢。

编辑:

我尝试了 Ketan在 AngularJs 中建议的 Chaining Ajax 调用

var someFn = function(id){
            Db.get(id, "abc")
                .then(function (d) {
                    console.log("At 1")
                    return Db.get(d.id, "def")
                }).then(function (d) {
                    console.log("At 2")
                    return d
                }, function (e) {
                    //error
                    return null;
                }).then(function (d) {
                    return d;
        });

        console.log("At 3")
    };

不过,如果我打电话给

var res = someFn(1)
console.log(res) /// undefined

chrome 终端显示At 2undefined. 我不确定为什么结果由someFnnot assigned to重新调整res

4

2 回答 2

4

你遇到困难的地方.then实际上并没有阻塞。它可以帮助您将同步代码转换为异步代码,但它不适合您。让我们从考虑您尝试重写的同步代码开始。想象一下Db.get是一个返回值而不是承诺的同步函数:

var someFn = function (id){
    try {
        var d = Db.get(id, "abc");
        console.log("At 1");
        var d = Db.get(d.id, "def");
        console.log("At 2")
        return d;
    } catch (ex) {
        console.log("At 3")
    }
};

在这种情况下,当我打电话时,someFn我得到一个值,而不是一个承诺。也就是说整个函数是同步的。

如果我们暂时快进几年并想象我们可以使用 ES6。这将使我们将您的函数重写为:

var someFn = $q.async(function* (id){
    try {
        var d = yield Db.get(id, "abc");
        console.log("At 1");
        var d = yield Db.get(d.id, "def");
        console.log("At 2")
        return d;
    } catch (ex) {
        console.log("At 3")
    }
});

这看起来很相似,但是这次我们Db.get返回了一个 Promise,并且someFn()也总是返回一个 Promise。关键字实际上“yield暂停”当前函数,直到实现承诺。这让它看起来就像同步代码,但它实际上是异步的。

回到现在,我们需要弄清楚如何写这个。调用的第二个参数.then是错误处理程序,因此与 ES6 示例完全等价的是:

var someFn = function (id){
    return Db.get(id, "abc")
        .then(function (d) {
            console.log("At 1");
            return Db.get(d.id, "def");
        })
        .then(function (d) {
            console.log("At 2");
            return d;
        })
        .then(null, function (ex) {
            console.log("At 3")
        });
});

请注意每个返回如何仅从其当前函数范围返回。没有办法强迫它一路跳出来someFn

另一个有趣的实验是:

Db.get('id', 'abc')
  .then(function () {
    console.log('B');
  });
console.log('A');

以上将始终记录:

A
B

因为.then不阻塞。

于 2013-05-01T02:58:56.870 回答
2

我在想然后会阻止,直到它得到承诺的回应

不,JS 中的 Promise 不是透明的、阻塞的 future,而只是一种链接回调的模式。承诺在回调执行之前返回 - 并在返回之后但在回调执行之前At 3记录。.then而且,如果您return在回调中,则外部someFn.

你宁愿使用类似的东西

var someFn = function(id){
    return Db.get(id, "abc")
      .then(function (d) {
        console.log("At 1")
        return Db.get(d.id, "def");
      })
      .then(function (d) {
        console.log("At 2")
        return d
      }, function (e) {
        //error
      });
}
someFn().then(function(d) {
    console.log("At 3")
});
于 2013-05-01T02:52:51.653 回答