在promise中超时函数的最佳通用做法是什么?

时间:2015-06-19 11:17:30

标签: javascript promise ecmascript-6 es6-promise

使用超时宣传功能调用

我已经看到很多资源提供了使用Promise.race在给定时间内超时函数调用的类似示例。这是Promise.race在实践中如何使用的一个很好的例子。这是一些示例代码:

function doWithinInterval(func, timeout) {
    var promiseTimeout = new Promise(function (fulfill, reject) {
       // Rejects as soon as the timeout kicks in
       setTimeout(reject, timeout);
    });
    var promiseFunc = new Promise(function (fulfill, reject) {
        var result = func(); // Function that may take long to finish
        // Fulfills when the given function finishes
        fulfill(result);
    });

    return Promise.race([promiseTimeout, promiseFunc]);
}

上面使用Promise.race的简单方法会在func完成之前一旦超时启动就拒绝承诺。否则,一旦func函数在超时间隔之前完成,项目就会完成。

这听起来不错且易于使用。

但是,这是在Promise中使用超时的最佳做法吗?

当然,如果我们想使用Promises为函数调用设置超时,可以使用上面的方法。这些行动似乎仍然是一个很好的承诺。但是,这被认为是在Promise中使用超时的好方法吗?如果没有,使用它的缺点是什么?

我一直在寻找其他方法,但无法找到原生的Promise方法。

相反,一些外部Promise库提供timeout功能,如下所示:

但是,Promise.timeout()似乎不是标准ECMAScript 6 API的一部分(如果我错了,请纠正我)。是否有任何推荐的方法来处理ES6 Promises原生的超时?

2 个答案:

答案 0 :(得分:12)

这取决于超时的含义。

如果您希望该功能停止,则为。

如果你只是想停止等待它,那么是(快速在ES6中掀起):

var wait = ms => new Promise(resolve => setTimeout(resolve, ms));
var timeout = (p, ms) => Promise.race([p, wait(ms).then(() => {
    throw new Error("Timeout after " + ms + " ms");
})]);

var wait = ms => new Promise(resolve => setTimeout(resolve, ms));
var timeout = (p, ms) => Promise.race([p, wait(ms).then(() => {
  throw new Error("Timeout after " + ms + " ms");
})]);

// Example:

var log = msg => div.innerHTML += "<p>" + msg + "</p>";
var failed = e => log(e.toString() + ", line " + e.lineNumber);

log("Waiting 5 seconds...");
timeout(wait(5000), 2000)
.then(() => log("...Done."))
.catch(failed);
<div id="div"></div>

如果你想取消操作(让它停止),那么希望该操作附带一个API来取消它,你应该使用它,因为ES6的承诺不是一个控制界面

Cancelable promises在ES6中是一个备受争议的话题,但提到的一些图书馆确实提供了这个概念。

答案 1 :(得分:8)

本机Promise.race方法在实际的promise完成后不会清除超时保证的计时器,因此进程将等到超时保证也完成。这意味着如果您将超时设置为1h并且我们的承诺在1分钟后完成,则该过程将在退出之前等待59分钟。

请改用此方法:

export function race({promise, timeout, error}) {
  let timer = null;

  return Promise.race([
    new Promise((resolve, reject) => {
      timer = setTimeout(reject, timeout, error);
      return timer;
    }),
    promise.then((value) => {
      clearTimeout(timer);
      return value;
    })
  ]);
}