2019-03-05 Promise/A+

Promises/A+

为实现者提供一个健全的、可互操作的 JavaScript promise 的开放标准。

一个 promise 代表了一个异步操作的最终结果。与一个 promise 交互的主要方式是通过它的 then 方法向 promise 注册回调函数(callbacks),回调函数则用来接收 promise 的的最终值(eventual value)或者promise失败(cannot be fulfilled)的原因。

本规范(specification )详细描述了 then方法的行为,提供了一个可互操作(interoperable)的基础,所有与Promises/A+ 规范一致的promise实现, 都可以依赖于这个基础。因此,规范应该被认为是非常稳定的。尽管Promises/A+组织可能偶尔会进行小的且向后兼容的更改去修改(revise)此规范,以解决新发现的极端案例(corner cases),但只有在仔细考虑、讨论和测试之后,我们才会集成大的或向后不兼容的更改。

从历史上看,Promises/A+阐述了早起 Promises/A proposal的行为条款,将其扩展以覆盖实际(de facto)行为,并删除了不明确或有问题的部分。

最后,Promises/A+规范核心不解决如何create、fulfill或reject promise,取而代之是选择提供一个可互操作的then方法。未来有关配套规范的工作可能涉及到这些主题。

1. 术语

1.1. “promise”是一个具有then方法的对象或函数,其行为符合本规范。
1.2. “thenable”是一个定义了then方法的对象或函数。
1.3. “value”是任何合法的javascript值(包括undefined, a thenable, or a promise)。
1.4. “exception”是一个使用throw语句抛出异常的值。。
1.5. “reason”是一个表明promise被rejected原因的值。

2. 要求

2.1 Promise状态

一个promise必须处于以下三种状态之一:pending,fulfilled 或 rejected。

2.1.1. 当处于 pending 状态时,a promise:
2.1.1.1. 即可以转换为 fulfilled 状态, 也可以转换 rejected 状态。

2.1.2. 当处于 fulfilled 状态时,a promise:
2.1.2.1. 一定不得转换到任何其他状态。
2.1.2.2. 必须有一个 value,且值不可变。

2.1.3. 当处于 rejected 状态时,a promise:
2.1.3.1. 一定不得转换到任何其他状态。
2.1.3.2. 必须有一个 reason ,且值不可变。

这里,“值不可变”是指不变恒等式(immutable identity)(如: ===),但并不意味着深层不变。

2.2 then方法

一个promise必须提供一个 then 方法用于存取当前或最终值(value或reason)。

一个promise的 then 方法接收两个参数:
promise.then(onFulfilled, onRejected)

2.2.1. onFulfilledonRejected 都是可选参数:
2.2.1.1. 如果 onFulFilled 不是函数,必须忽略它。
2.2.1.2. 如果onRejected 不是函数,必须忽略它。

2.2.2. 如果 onFulfilled 是一个函数:
2.2.2.1. 此函数必须在 promise 变为 fulfilled 之后被调用,并以 promisevalue 作为第一个参数。
2.2.2.2. 此函数不得在 promise 变为 fulfilled 之前被调用。
2.2.2.3. 此函数被调用的次数不得超过一次。

2.2.3. 如果 onRejected 是一个函数:
2.2.3.1. 此函数必须在 promise 变为 rejected 之后被调用,并以 promise 的 reason 作为第一个参数。
2.2.3.2. 此函数不得在 promise 变为rejected 之前被调用。
2.2.3.3. 此函数被调用的次数不得超过一次。

2.2.4. onFulfilledonRejected 直到执行上下文(execution context)堆栈只包含平台代码时才可以被调用。[3.1]。

2.2.5. onFulfilledonRejected 必须以函数方式被调用,且没有this值(译者注: 以函数方式调用是指onFulfilled(v)这种方式,不可用 new onFulfilled() 方式或 onFulfilled.call() / onFulfilled.apply() 方式)。[3.2]

2.2.6. 在同一个promise上 then 方法可以多次被调用 。
2.2.6.1. 如果/当 promise 变为或已处于 fulfilled 时,各自的 onFulfilled 回调必须以其在 then 中注册顺序来执行。
2.2.6.2. 如果/当 promise 变为或已处于 rejected 时,各自的 onRejected 回调必须以其在 then 中注册顺序来执行。

2.2.7. then 方法必须返回一个promise。 [3.3].
promise2 = promise1.then(onFulfilled, onRejected);
2.2.7.1. 如果 onFulfilledonRejected 返回了一个值 x, 则执行Promise Resolution Procedure[[Resolve]](promise2, x)
2.2.7.2. 如果 onFulfilledonRejected 抛出了一个异常 e,则 promise2 必须被rejected,且以 e 作为reason。
2.2.7.3. 如果 onFulfilled 不是函数且 promise1 是 onFulfilled 状态,则 promise2 必须被Fulfilled, 且和promise1有同一个value。
2.2.7.4. 如果 onRejected 不是函数且 promise1 已处于 rejected 状态,则 promise2 必须 被 rejected, 且和promise1有同的 reason。

2.3 Promise决议程序(Resolution Procedure)

Promise Resolution Procedure 是以一个promisevalue为参数的抽象操作,我们将其表示为 [[Resolve]](promise, x)。如果 x 是一个 thenable , 前提是x的行为至少有点像promise时,Promise Resolution Procedure 会试图让 promise 采用 x 的 state。与此不同,promise 会被Promise Resolution Procedure以 x 为值 fulfilled。

thenable 的这种处理允许promise实现相互操作,只要它们公开了一个暴露了符合Promises/A+规范 的 then 方法即可。这种方式也使得遵循 Promises/A+规范 的实现可以“同化”那些没有遵循 Promises/A+规范 但具备合理的 then 方法的promise实现。
它还允许Promises/A+实现使用合理的II方法“同化”不合格的实现。

执行 [[Resolve]](promise, x) 需运行以下步骤:

2.3.1. 如果 promise 和 x 指向同一个对象,则 reject promise 并且以 TypeError 作为 reason。

2.3.2. 如果 x 是一个promise,则采用 x 的状态 [3.4]:
2.3.2.1. 如果 x 处于 pending 状态,则 promise 必须保持 pending 状态直到 x 被 fulfilled 或 rejected。
2.3.2.2. 当 x 状态是 fulfilled 时,用相同的 value 来 fulfill promise。
2.3.2.3. 当 x 状态是 rejected 时,用相同的 reason 来 reject promise。

2.3.3. 如果 x 是一个对象或函数,
2.3.3.1. 令 then 指向 x.then。[3.5]
2.3.3.2. 如果x.then导致抛出异常 e,则以 e 为 reason 来 reject promise。
2.3.3.3. 如果 then 是一个函数,则x.then调用它,,以 resolvePromise 作为第一个参数,以 rejectPromise 作为第二个参数,情况如下:
2.3.3.3.1. 如果/当 resolvePromise被调用, 且value是 y,则执行 [[Resolve]](promise, y)。
2.3.3.3.2. 如果/当 rejectPromise 被调用了且value是 r,则以 r 来 reject promise。
2.3.3.3.3. 如果 resolvePromise 和 rejectPromise 都被调用了,或者以相同参数被调用多次,则第一次调用优先(precedence),其余相关调用忽略。
2.3.3.3.4. 如果调用 then 抛出异常 e:
2.3.3.3.4.1 如果 resolvePromise 或 rejectPromise 已经被调用过了,则忽略这个异常。
2.3.3.3.4.2 否则,reject promise, 且reason是e。
2.3.3.4. 如果 then 不是函数, fulfill promise, 且value是x。

2.3.4. 如果 x 既不是对象也不是函数,则fulfill promise, 且value是x。

如果一个promise被一个 thenable 决议,且此 thenable 在一个循环的thenable链中,[[Resolve]](promise, thenable) 的递归特性会导致 [[Resolve]](promise, thenable) 再次被调用,根据上面的算法,这会导致无限递归。因此我们鼓励(但不强制要求)promise实现时能检测无限递归,并在出现无限递归时以一个 TypeError 来 reject promise。

3. 说明

3.1. 这里的“平台代码(platform code)”意为引擎、环境和实现promise的代码。在实践中,这个需求确保了onFulfilled and onRejected的异步执行,在调用事件循环之后,使用一个新的堆栈。这可以通过“宏任务”机制(如settimeout或setimmediate)或“微任务”机制(如mutationobserver或process.nextick)来实现。由于Promise实现被视为平台代码,因此它本身可能包含一个任务调度队列或“蹦床”,在其中调用处理程序。

3.2. 这里指的是在 严格模式(strict mode) 下,this 在 onFulfilled 和 onRejected 中值为 undefined;在 非严格模式 下,this 指向全局对象(global object)。

3.3.实现可以允许promise2==promise1,前提是实现满足所有要求。每个实现都应该形成文档, 它是否能够生成promise2==promise1以及在什么条件下。

3.4. 一般来说,只有当 x 来自当前的实例时,才知道它是一个真正的 promise。本条款允许使用特定于实现的方法来采用已知一致promise的状态。

3.5. 先存储x.then 的引用,然后测试此引用,再然后调用此引用的过程,避免了对 x.then 属性的多次访问。这些预防措施对于确保访问器属性的一致性非常重要,访问器属性的值可能在两次检索之间发生更改。

3.6. 实现不应该对 thenable 的深度设置任意限制,并且假定超过该任意限制,递归将是无限的。只有真正的循环才能导致类型错误;如果遇到无限多个不同的 thenable ,则永远递归是正确的行为。

https://promisesaplus.com/

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,014评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,796评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,484评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,830评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,946评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,114评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,182评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,927评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,369评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,678评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,832评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,533评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,166评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,885评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,128评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,659评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,738评论 2 351

推荐阅读更多精彩内容