javascript - 我如何判断一个对象是否是 jQuery Promise/Deferred?

标签 javascript jquery jquery-deferred

我有一个接受单个参数的函数。我需要能够判断此参数是 jQuery Promise 还是 Deferred 对象。如果不是,那么该值可能是任何类型并具有任何属性,因此仅仅存在 promise 方法是不安全的。

这是我希望我的函数如何运行的示例:

function displayMessage(message) {
  if (message is a Promise or Deferred) {
    message.then(displayMessage);
  } else {
    alert(message);
  }
}

注意 promise 的递归处理:如果一个 promise 用另一个 promise 值解决,我们不会显示它,我们等待它被解决。如果它返回另一个 promise ,请重复。


这很重要,因为如果不是这种情况,我只能使用 jQuery.when :

function displayMessage(message) {
  jQuery.when(message).then(function(messageString) {
    alert(messageString);
  });
}

这将正确处理值和值的 promise ...

displayMessage("hello");                            // alerts "hello"
displayMessage(jQuery.Deferred().resolve("hello")); // alerts "hello"

...但是一旦我们得到值(value) promise 的 promise ,它就会崩溃:

displayMessage(jQuery.Deferred().resolve(
  jQuery.Deferred().resolve("hello")
));                                                 // alerts "[object Object]"

jQuery.when 能够判断一个值是否为 promise,因此显然这是可能的。我该如何检查?

最佳答案

jQuery.when is able to tell if a value is promise, so apparently it is possible.

这是错误的。 jQuery 本身无法检查一个对象是否是完全准确的 promise 。如果您查看 jQuery.when 的来源in the jQuery source viewer你可以看到它所做的就是这样:

jQuery.isFunction(firstParam.promise)

如果您要返回的对象有自己的 .promise()方法,jQuery.when会行为不端:

var trickyValue = {
  promise: function() { return 3; },
  value: 2
};

jQuery.when(trickyValue).then(function(obj) {
  alert(obj.value);
});

这会抛出 TypeError: Object 3 has no method 'then' , 因为 jQuery 假定该对象是一个 promise 并且信任其 .promise() 的值方法。

这可能无法正确解决。 promise 对象在 jQuery.Deferred 中创建为对象字面量(view source)。它没有原型(prototype),也没有任何其他可用于区分它的真正独特的属性。

但是,我可以想到一个 hacky 解决方案,只要只使用一个版本的 jQuery,它就应该是可靠的:

function isPromise(value) {
  if (typeof value === 'object' && typeof value.then !== "function") {
    return false;
  }
  var promiseThenSrc = String($.Deferred().then);
  var valueThenSrc = String(value.then);
  return promiseThenSrc === valueThenSrc;
}

isPromise("test");                 // false
isPromise($.Deferred());           // true
isPromise($.Deferred().promise()); // true

将一个函数转换成一个字符串可以得到它的源代码,所以在这里我比较了 .then 的源代码新方法Deferred反对我感兴趣的值(value)。你的值(value)不会有 .thenjQuery.Deferred 具有完全相同源代码的方法或 Promise 1.

1. 除非您在恶劣的环境中运行,否则您应该放弃。


如果您对 jQuery promises 不是特别感兴趣,但想检测 任何 类型的 Promise,包括 ECMAScript 6 中的内置 Promise,您可以测试 value 是否是一个对象并且具有一个then方法:

if (typeof value === 'object' && typeof value.then === 'function') {
  // handle a promise
} else {
  // handle a concrete value
}

这是 ES6 中定义的几个 Promise 处理函数的方法。你可以看到这个描述 in the specification of the resolve(...) functions ,部分引用如下:

When a promise resolve function F is called with argument resolution, the following steps are taken:

[...]

  1. If Type(resolution) is not Object, then
    1. Return FulfillPromise(promise, resolution).
  2. Let then be Get(resolution, "then").
  3. If then is an abrupt completion, then
    1. Return RejectPromise(promise, then.[[value]]).
  4. Let thenAction be then.[[value]].
  5. If IsCallable(thenAction) is false, then
    1. Return FulfillPromise(promise, resolution).
  6. Perform EnqueueJob ("PromiseJobs", PromiseResolveThenableJob, «‍promise, resolution, thenAction»)

关于javascript - 我如何判断一个对象是否是 jQuery Promise/Deferred?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/13075592/

相关文章:

javascript - 切换到 JS Defer 转换 jquery ajax 重新加载整个页面

javascript - 为什么 AJAX 的响应为 "_blank"对第一个函数起作用,但对第二个函数不起作用?

javascript - 如何将元素的背景颜色设置为该元素的类名的值?

javascript - 用于 TypeScript : Class or Interface? 中的纯数据对象的内容

javascript - 页面元素飞来飞去

jquery - 自动完成文本框----未找到值

javascript - 带有递归 Ajax 的 Jquery Promise

javascript - JavaScript 中的高级表单验证

javascript - 在 Vue 中离开页面之前如何警告用户未保存的更改