angularjs 等待所有promise解决

1tu0hz3e  于 2023-03-22  发布在  Angular
关注(0)|答案(6)|浏览(163)

所以我有一个情况,我有多个未知长度的promise链。我想在所有的链都被处理完后运行一些动作。这可能吗?下面是一个例子:

app.controller('MainCtrl', function($scope, $q, $timeout) {
    var one = $q.defer();
    var two = $q.defer();
    var three = $q.defer();

    var all = $q.all([one.promise, two.promise, three.promise]);
    all.then(allSuccess);

    function success(data) {
        console.log(data);
        return data + "Chained";
    }

    function allSuccess(){
        console.log("ALL PROMISES RESOLVED")
    }

    one.promise.then(success).then(success);
    two.promise.then(success);
    three.promise.then(success).then(success).then(success);

    $timeout(function () {
        one.resolve("one done");
    }, Math.random() * 1000);

    $timeout(function () {
        two.resolve("two done");
    }, Math.random() * 1000);

    $timeout(function () {
        three.resolve("three done");
    }, Math.random() * 1000);
});

在这个例子中,我为promises 1,2和3设置了一个$q.all(),它们将在某个随机时间被解析。然后我将promises添加到1和3的末尾。我希望当所有链都被解析时,all也会被解析。下面是我运行这段代码时的输出:

one done 
one doneChained
two done
three done
ALL PROMISES RESOLVED
three doneChained
three doneChainedChained

有没有办法等锁链解开?

vwoqyblh

vwoqyblh1#

我希望在所有的锁链都解开的时候,一切都能解开。
当然,然后只需将每条链的promise传递到all()中,而不是最初的promise:

$q.all([one.promise, two.promise, three.promise]).then(function() {
    console.log("ALL INITIAL PROMISES RESOLVED");
});

var onechain   = one.promise.then(success).then(success),
    twochain   = two.promise.then(success),
    threechain = three.promise.then(success).then(success).then(success);

$q.all([onechain, twochain, threechain]).then(function() {
    console.log("ALL PROMISES RESOLVED");
});
vu8f3i0k

vu8f3i0k2#

accepted answer是正确的。我想提供一个例子来详细说明一下,对那些不熟悉promise的人。

示例:

在我的示例中,我需要在呈现内容之前用不同的镜像URL替换img标记的src属性(如果可用)。

var img_tags = content.querySelectorAll('img');

function checkMirrorAvailability(url) {

    // blah blah 

    return promise;
}

function changeSrc(success, y, response) {
    if (success === true) {
        img_tags[y].setAttribute('src', response.mirror_url);
    } 
    else {
        console.log('No mirrors for: ' + img_tags[y].getAttribute('src'));
    }
}

var promise_array = [];

for (var y = 0; y < img_tags.length; y++) {
    var img_src = img_tags[y].getAttribute('src');

    promise_array.push(
        checkMirrorAvailability(img_src)
        .then(

            // a callback function only accept ONE argument. 
            // Here, we use  `.bind` to pass additional arguments to the
            // callback function (changeSrc).

            // successCallback
            changeSrc.bind(null, true, y),
            // errorCallback
            changeSrc.bind(null, false, y)
        )
    );
}

$q.all(promise_array)
.then(
    function() {
        console.log('all promises have returned with either success or failure!');
        render(content);
    }
    // We don't need an errorCallback function here, because above we handled
    // all errors.
);

说明:

AngularJS docs
then方法:

then(successCallback,errorCallback,notifyCallback)-无论promise何时被解析或拒绝,只要结果可用,就异步调用success或error中的一个回调。调用回调时使用单参数:结果或拒绝原因。

$q.all(promises)
将多个promise合并为一个promise,当所有输入promise都被解析时,该promise将被解析。
promises参数可以是一个promise数组。
关于bind(),更多信息在这里:https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Function/bind

wtzytmuj

wtzytmuj3#

最近有这个问题,但与未知数量的承诺。解决使用jQuery.map()

function methodThatChainsPromises(args) {

    //var args = [
    //    'myArg1',
    //    'myArg2',
    //    'myArg3',
    //];

    var deferred = $q.defer();
    var chain = args.map(methodThatTakeArgAndReturnsPromise);

    $q.all(chain)
    .then(function () {
        $log.debug('All promises have been resolved.');
        deferred.resolve();
    })
    .catch(function () {
        $log.debug('One or more promises failed.');
        deferred.reject();
    });

    return deferred.promise;
}
nbnkbykc

nbnkbykc5#

您可以使用"await" in an "async function"

app.controller('MainCtrl', async function($scope, $q, $timeout) {
  ...
  var all = await $q.all([one.promise, two.promise, three.promise]); 
  ...
}

注意:我不能100%确定你可以从一个非异步函数调用一个异步函数并得到正确的结果。
也就是说,这永远不会在网站上使用。但对于负载测试/集成测试......可能会。
示例代码:

async function waitForIt(printMe) {
  console.log(printMe);
  console.log("..."+await req());
  console.log("Legendary!")
}

function req() {
  
  var promise = new Promise(resolve => {
    setTimeout(() => {
      resolve("DARY!");
    }, 2000);
    
  });

    return promise;
}

waitForIt("Legen-Wait For It");
00jrzges

00jrzges6#

我在ocLazyLoad库中使用了这个例子:

var promises = [];
angular.forEach(paths, function loading(path) {
    promises.push(buildElement('css', path, params));
});
$q.all(promises).then(function success() {
    callback();
 }, function error(err) {
        callback(err);
  });

相关问题