8

我很难按顺序运行承诺。

var getDelayedString = function(string) {
    var deferred = Q.defer();

    setTimeout(function() {
        document.write(string+" ");
        deferred.resolve();
    }, 500);

    return deferred.promise;
};

var onceUponATime = function() {
    var strings = ["Once", "upon", "a", "time"];

    var promiseFuncs = [];

    strings.forEach(function(str) {
        promiseFuncs.push(getDelayedString(str));
    });

    //return promiseFuncs.reduce(Q.when, Q());
    return promiseFuncs.reduce(function (soFar, f) {
        return soFar.then(f);
    }, Q());    
};

getDelayedString("Hello")
.then(function() {
    return getDelayedString("world!")
})
.then(function() {
    return onceUponATime();
})
.then(function() {
    return getDelayedString("there was a guy and then he fell.")
})
.then(function() {
    return getDelayedString("The End!")
})

onceUponATime() 应该顺序输出 ["Once", "on", "a", "time"] 但由于某种原因它们会立即输出。

jsFiddle在这里:http: //jsfiddle.net/6Du42/2/

知道我做错了什么吗?

4

1 回答 1

15

但由于某种原因,它们会立即被输出。

你已经在这里给他们打电话了:

promiseFuncs.push(getDelayedString(str));
//                                ^^^^^

你需要推动function(){ return getDelayedString(str); }. 顺便说一句,与其在循环中使用推送到数组,each不如使用map. 实际上,您实际上并不需要它,但可以直接reduce通过strings数组:

function onceUponATime() {
    var strings = ["Once", "upon", "a", "time"];

    return strings.reduce(function (soFar, s) {
        return soFar.then(function() {
            return getDelayedString(s);
        });
    }, Q());    
}

哦,不要使用document.write.

于 2013-08-22T17:48:52.637 回答