浅析Promise原理

发布时间:2025-05-24 17:30:23 作者:益华网络 来源:undefined 浏览量(1) 点赞(1)
摘要:一、Promise基础用法 1.1 基本用法 new Promise( function( resolve, reject) {

一、Promise基础用法

1.1 基本用法

new Promise( function( resolve, reject) {     //待处理的异步逻辑     //处理结束后,调用resolve或reject方法 }) 新建一个 promise很简单,只需要 new一个 promise对象即可。所以 promise本质上就是一个函数,它接受一个函数作为参数,并且会返回 promise对象,这就给链式调用提供了基础 其实 Promise函数的使命,就是构建出它的实例,并且负责帮我们管理这些实例。而这些实例有以下三种状态: pending: 初始状态,位履行或拒绝 fulfilled: 意味着操作成功完成 rejected: 意味着操作失败

pending 状态的  Promise对象可能以  fulfilled状态返回了一个值,也可能被某种理由(异常信息)拒绝( reject)了。当其中任一种情况出现时, Promise 对象的  then 方法绑定的处理方法(handlers)就会被调用,then方法分别指定了 resolve方法和 reject方法的回调函数

var promise = new Promise( function( resolve, reject) {   if ( /* 异步操作成功 */){    resolve(value);  } else {    reject(error);  } }); promise.then( function( value) {   // 如果调用了resolve方法,执行此函数 }, function( value) {   // 如果调用了reject方法,执行此函数 });

上述代码很清晰的展示了 promise对象运行的机制。下面再看一个示例:

var getJSON = function( url) {   var promise = new Promise( function( resolve, reject){     var client = new XMLHttpRequest();    client.open( "GET", url);

   client.>

   client.responseType = "json";    client.setRequestHeader( "Accept", "application/json");    client.send();     function handler( ) {       if ( this.status === 200) {              resolve( this.response);          } else {              reject( new Error( this.statusText));          }    };  });   return promise; }; getJSON( "/posts.json").then( function( json) {   console.log( Contents: + json); }, function( error) {   console.error( 出错了, error); });

上面代码中, resolve方法和 reject方法调用时,都带有参数。它们的参数会被传递给回调函数。 reject方法的参数通常是 Error对象的实例,而 resolve方法的参数除了正常的值以外,还可能是另一个 Promise实例,比如像下面这样。

var p1 = new Promise( function( resolve, reject){   // ... some code }); var p2 = new Promise( function( resolve, reject){   // ... some code  resolve(p1); })

上面代码中, p1和 p2都是 Promise的实例,但是 p2的 resolve方法将 p1作为参数,这时 p1的状态就会传递给 p2。如果调用的时候, p1的状态是 pending,那么 p2的回调函数就会等待 p1的状态改变;如果 p1的状态已经是 fulfilled或者 rejected,那么 p2的回调函数将会立刻执行

1.2 promise捕获错误

Promise.prototype.catch方法是 Promise.prototype.then(null, rejection)的别名,用于指定发生错误时的回调函数

getJSON( "/visa.json").then( function( result) {   // some code }).catch( function( error) {   // 处理前一个回调函数运行时发生的错误   console.log( 出错啦!, error); });

Promise对象的错误具有“冒泡”性质,会一直向后传递,直到被捕获为止。也就是说,错误总是会被下一个 catch语句捕获

getJSON( "/visa.json").then( function( json) {   return json.name; }).then( function( name) {   // proceed }).catch( function( error) {     //处理前面任一个then函数抛出的错误 });

1.3 常用的promise方法

Promise.all方法

Promise.all方法用于将多个 Promise实例,包装成一个新的 Promise实例

var p = Promise.all([p1,p2,p3]); 上面代码中, Promise.all方法接受一个数组作为参数, p1、 p2、 p3都是 Promise对象的实例。( Promise.all方法的参数不一定是数组,但是必须具有 iterator接口,且返回的每个成员都是 Promise实例。)

p的状态由 p1、 p2、 p3决定,分成两种情况

只有 p1、 p2、 p3的状态都变成 fulfilled, p的状态才会变成 fulfilled,此时 p1、 p2、 p3的返回值组成一个数组,传递给 p的回调函数 只要 p1、 p2、 p3之中有一个被 rejected, p的状态就变成 rejected,此时第一个被 reject的实例的返回值,会传递给p的回调函数 // 生成一个Promise对象的数组 var promises = [ 2, 3, 5, 7, 11, 13].map( function( id){   return getJSON( "/get/addr" + id + ".json"); }); Promise.all(promises).then( function( posts) {   // ...   }).catch( function( reason){   // ... });

Promise.race方法

Promise.race方法同样是将多个 Promise实例,包装成一个新的 Promise实例。

var p = Promise.race([p1,p2,p3]);

上面代码中,只要 p1、 p2、 p3之中有一个实例率先改变状态,p的状态就跟着改变。那个率先改变的Promise实例的返回值,就传递给p的返回值

如果 Promise.all方法和 Promise.race方法的参数,不是 Promise实例,就会先调用下面讲到的 Promise.resolve方法,将参数转为 Promise实例,再进一步处理

Promise.resolve

有时需要将现有对象转为 Promise对象, Promise.resolve方法就起到这个作用

var jsPromise = Promise.resolve($.ajax( /whatever.json));

上面代码将 jQuery生成 deferred对象,转为一个新的 ES6的 Promise对象

如果 Promise.resolve方法的参数,不是具有 then方法的对象(又称 thenable对象),则返回一个新的 Promise对象,且它的状态为 fulfilled。 var p = Promise.resolve( Hello); p.then( function ( s){   console.log(s) }); // Hello 上面代码生成一个新的 Promise对象的实例 p,它的状态为 fulfilled,所以回调函数会立即执行, Promise.resolve方法的参数就是回调函数的参数 如果 Promise.resolve方法的参数是一个 Promise对象的实例,则会被原封不动地返回 Promise.reject(reason)方法也会返回一个新的 Promise实例,该实例的状态为 rejected。 Promise.reject方法的参数 reason,会被传递给实例的回调函数 var p = Promise.reject( 出错啦); p.then( null, function ( error){   console.log(error) }); // 出错了

1.4 Async/await简化写法

function getDataAsync ( url) {     return new Promise( ( resolve, reject) => {        setTimeout( () => {             var res = {                url: url,                data: Math.random()            }            resolve(res)        }, 1000)    }) } async function getData ( ) {     var res1 = await getDataAsync( /page/1?param=123)     console.log(res1)     var res2 = await getDataAsync( `/page/2?param= ${res1.data}`)     console.log(res2)     var res3 = await getDataAsync( `/page/2?param= ${res2.data}`)     console.log(res3) }

async/await 是基于  Promise 的,因为使用  async 修饰的方法最终返回一个  Promise, 实际上, async/await 可以看做是使用  Generator 函数处理异步的语法糖,我们来看看如何使用  Generator 函数处理异步

1.5 Generator

首先异步函数依然是:

function getDataAsync ( url) {     return new Promise( ( resolve, reject) => {        setTimeout( () => {             var res = {                url: url,                data: Math.random()            }            resolve(res)        }, 1000)    }) }

使用  Generator 函数可以这样写

function * getData ( ) {     var res1 = yield getDataAsync( /page/1?param=123)     console.log(res1)     var res2 = yield getDataAsync( `/page/2?param= ${res1.data}`)     console.log(res2)     var res3 = yield getDataAsync( `/page/2?param= ${res2.data}`)     console.log(res3)) }

然后我们这样逐步执行

var g = getData() g.next().value.then( res1 => {    g.next(res1).value.then( res2 => {        g.next(res2).value.then( () => {            g.next()        })    }) })

上面的代码,我们逐步调用遍历器的  next() 方法,由于每一个  next() 方法返回值的  value 属性为一个  Promise 对象,所以我们为其添加  then 方法, 在  then方法里面接着运行  next 方法挪移遍历器指针,直到  Generator 函数运行完成,实际上,这个过程我们不必手动完成,可以封装成一个简单的执行器

function run ( gen) {     var g = gen()     function next ( data) {         var res = g.next(data)         if (res.done) return res.value        res.value.then( ( data) => {            next(data)        })    }    next() }

run方法用来自动运行异步的  Generator 函数,其实就是一个递归的过程调用的过程。这样我们就不必手动执行  Generator 函数了。 有了  run 方法,我们只需要这样运行 getData 方法

run(getData)

这样,我们就可以把异步操作封装到  Generator 函数内部,使用  run 方法作为  Generator 函数的自执行器,来处理异步。其实我们不难发现,  async/await 方法相比于  Generator 处理异步的方式,有很多相似的地方,只不过  async/await 在语义化方面更加明显,同时  async/await 不需要我们手写执行器,其内部已经帮我们封装好了,这就是为什么说  async/await 是  Generator 函数处理异步的语法糖了

二、Promise实现原理剖析

2.1 Promise标准

Promise 规范有很多,如 Promise/A, Promise/B, Promise/D以及  Promise/A 的升级版  Promise/A+。 ES6中采用了  Promise/A+ 规范

中文版规范:  Promises/A+规范(中文)

Promise标准解读

一个 promise的当前状态只能是 pending、 fulfilled和 rejected三种之一。状态改变只能是 pending到 fulfilled或者 pending到 rejected。状态改变不可逆 promise的 then方法接收两个可选参数,表示该 promise状态改变时的回调( promise.then(onFulfilled, onRejected))。 then方法返回一个 promise。 then 方法可以被同一个  promise 调用多次

2.2 实现Promise

构造函数

function Promise( resolver) {}

原型方法

Promise.prototype.then = function( ) {} Promise.prototype.catch = function( ) {}

静态方法

Promise.resolve = function( ) {} Promise.reject = function( ) {} Promise.all = function( ) {} Promise.race = function( ) {}

2.3 极简promise雏形

function Promise( fn) {     var value = null,        callbacks = [];   //callbacks为数组,因为可能同时有很多个回调     this.then = function ( onFulfilled) {        callbacks.push(onFulfilled);    };     function resolve( value) {        callbacks.forEach( function ( callback) {            callback(value);        });    }    fn(resolve); }

大致的逻辑是这样的

调用 then方法,将想要在 Promise异步操作成功时执行的回调放入 callbacks队列,其实也就是注册回调函数,可以向观察者模式方向思考 创建 Promise实例时传入的函数会被赋予一个函数类型的参数,即 resolve,它接收一个参数 value,代表异步操作返回的结果,当一步操作执行成功后,用户会调用 resolve方法,这时候其实真正执行的操作是将 callbacks队列中的回调一一执行 //例1 function getUserId( ) {     return new Promise( function( resolve) {         //异步请求        http.get(url, function( results) {            resolve(results.id)        })    }) } getUserId().then( function( id) {     //一些处理 }) // 结合例子1分析 // fn 就是getUserId函数 function Promise( fn) {     var value = null,        callbacks = [];   //callbacks为数组,因为可能同时有很多个回调     // 当用户调用getUserId().then的时候开始注册传进来的回调函数     // onFulfilled就是例子中的function(id){}     // 把then的回调函数收集起来 在resolve的时候调用     this.then = function ( onFulfilled) {        callbacks.push(onFulfilled);    };     // value是fn函数执行后返回的值     function resolve( value) {         // callbacks是传给then的回调函数就是例子中的function(id){}         // 遍历用户通过then传递进来的回调函数把resolve成功的结果返回给then调用即then(function(data){ console.log(data) }) 这里的data就是通过这里调用返回        callbacks.forEach( function ( callback) {            callback(value);        });    }     //执行fn函数即getUserId()并且传入函数参数resolve 当fn执行完成返回的值传递给resolve函数    fn(resolve); }

结合例1中的代码来看,首先 new Promise时,传给 promise的函数发送异步请求,接着调用 promise对象的 then属性,注册请求成功的回调函数,然后当异步请求发送成功时,调用 resolve(results.id)方法, 该方法执行 then方法注册的回调数组

then方法应该能够链式调用,但是上面的最基础简单的版本显然无法支持链式调用。想让 then方法支持链式调用,其实也是很简单的 this.then = function ( onFulfilled) {    callbacks.push(onFulfilled);     return this; };

只要简单一句话就可以实现类似下面的链式调用

// 例2 getUserId().then( function ( id) {     // 一些处理 }).then( function ( id) {     // 一些处理 });

2.4 加入延时机制

上述代码可能还存在一个问题:如果在 then方法注册回调之前, resolve函数就执行了,怎么办?比如 promise内部的函数是同步函数

// 例3 function getUserId( ) {     return new Promise( function ( resolve) {        resolve( 9876);    }); } getUserId().then( function ( id) {     // 一些处理 });

这显然是不允许的, Promises/A+规范明确要求回调需要通过异步方式执行,用以保证一致可靠的执行顺序。因此我们要加入一些处理,保证在 resolve执行之前, then方法已经注册完所有的回调。我们可以这样改造下 resolve函数:

function resolve( value) {    setTimeout( function( ) {        callbacks.forEach( function ( callback) {            callback(value);        });    }, 0) }

上述代码的思路也很简单,就是通过 setTimeout机制,将 resolve中执行回调的逻辑放置到 JS任务队列末尾,以保证在 resolve执行时, then方法的回调函数已经注册完成

但是,这样好像还存在一个问题,可以细想一下:如果 Promise异步操作已经成功,这时,在异步操作成功之前注册的回调都会执行,但是在 Promise异步操作成功这之后调用的 then注册的回调就再也不会执行了,这显然不是我们想要的

2.5 加入状态

我们必须加入状态机制,也就是大家熟知的 pending、 fulfilled、 rejected

Promises/A+规范中的 2.1 Promise States中明确规定了, pending可以转化为 fulfilled或 rejected并且只能转化一次,也就是说如果 pending转化到 fulfilled状态,那么就不能再转化到r ejected。并且 fulfilled和 rejected状态只能由 pending转化而来,两者之间不能互相转换

//改进后的代码是这样的: function Promise( fn) {     var state = pending,        value = null,        callbacks = [];     this.then = function ( onFulfilled) {         if (state === pending) {            callbacks.push(onFulfilled);             return this;        }        onFulfilled(value);         return this;    };     function resolve( newValue) {        value = newValue;        state = fulfilled;        setTimeout( function ( ) {            callbacks.forEach( function ( callback) {                callback(value);            });        }, 0);    }    fn(resolve); }

上述代码的思路是这样的: resolve执行时,会将状态设置为 fulfilled,在此之后调用 then添加的新回调,都会立即执行

2.6 链式Promise

如果用户在 then函数里面注册的仍然是一个 Promise,该如何解决?比如下面的例4

// 例4 getUserId()    .then(getUserJobById)    .then( function ( job) {         // 对job的处理    }); function getUserJobById( id) {     return new Promise( function ( resolve) {        http.get(baseUrl + id, function( job) {            resolve(job);        });    }); } 这种场景相信用过 promise的人都知道会有很多,那么类似这种就是所谓的链式 Promise 链式 Promise是指在当前 promise达到 fulfilled状态后,即开始进行下一个 promise(后邻 promise)。那么我们如何衔接当前 promise和后邻 promise呢?(这是这里的难点 只要在 then方法里面 return一个 promise就好啦。 Promises/A+规范中的 2.2.7就是这样

下面来看看这段暗藏玄机的 then方法和 resolve方法改造代码

function Promise( fn) {     var state = pending,        value = null,        callbacks = [];     this.then = function ( onFulfilled) {         return new Promise( function ( resolve) {            handle({                onFulfilled: onFulfilled || null,                resolve: resolve            });        });    };     function handle( callback) {         if (state === pending) {            callbacks.push(callback);             return;        }         //如果then中没有传递任何东西         if(!callback.onFulfilled) {            callback.resolve(value);             return;        }         var ret = callback.onFulfilled(value);        callback.resolve(ret);    }     function resolve( newValue) {         if (newValue && ( typeof newValue === object || typeof newValue === function)) {             var then = newValue.then;             if ( typeof then === function) {                then.call(newValue, resolve);                 return;            }        }        state = fulfilled;        value = newValue;        setTimeout( function ( ) {            callbacks.forEach( function ( callback) {                handle(callback);            });        }, 0);    }    fn(resolve); }

我们结合例4的代码,分析下上面的代码逻辑,为了方便阅读,我把例4的代码贴在这里

// 例4 getUserId()    .then(getUserJobById)    .then( function ( job) {         // 对job的处理    }); function getUserJobById( id) {     return new Promise( function ( resolve) {        http.get(baseUrl + id, function( job) {            resolve(job);        });    }); } then方法中,创建并返回了新的 Promise实例,这是串行 Promise的基础,并且支持链式调用 handle方法是 promise内部的方法。 then方法传入的形参 onFulfilled以及创建新 Promise实例时传入的 resolve均被 push到当前 promise的 callbacks队列中,这是衔接当前 promise和后邻 promise的关键所在 getUserId生成的 promise(简称 getUserId promise)异步操作成功,执行其内部方法 resolve,传入的参数正是异步操作的结果 id 调用 handle方法处理 callbacks队列中的回调: getUserJobById方法,生成新的 promise(g etUserJobById promise) 执行之前由 getUserId promise的 then方法生成的新 promise(称为 bridge promise)的 resolve方法,传入参数为 getUserJobById promise。这种情况下,会将该 resolve方法传入 getUserJobById promise的 then方法中,并直接返回 在 getUserJobById promise异步操作成功时,执行其 callbacks中的回调: getUserId bridge promise中的 resolve方法 最后执行 getUserId bridge promise的后邻 promise的 callbacks中的回调

2.7 失败处理

在异步操作失败时,标记其状态为 rejected,并执行注册的失败回调

//例5 function getUserId( ) {     return new Promise( function( resolve) {         //异步请求        http.get(url, function( error, results) {             if (error) {                reject(error);            }            resolve(results.id)        })    }) } getUserId().then( function( id) {     //一些处理 }, function( error) {     console.log(error) })

有了之前处理 fulfilled状态的经验,支持错误处理变得很容易,只需要在注册回调、处理状态变更上都要加入新的逻辑

function Promise( fn) {     var state = pending,        value = null,        callbacks = [];     this.then = function ( onFulfilled, onRejected) {         return new Promise( function ( resolve, reject) {            handle({                onFulfilled: onFulfilled || null,                onRejected: onRejected || null,                resolve: resolve,                reject: reject            });        });    };     function handle( callback) {         if (state === pending) {            callbacks.push(callback);             return;        }         var cb = state === fulfilled ? callback.onFulfilled : callback.onRejected,            ret;         if (cb === null) {            cb = state === fulfilled ? callback.resolve : callback.reject;            cb(value);             return;        }        ret = cb(value);        callback.resolve(ret);    }     function resolve( newValue) {         if (newValue && ( typeof newValue === object || typeof newValue === function)) {             var then = newValue.then;             if ( typeof then === function) {                then.call(newValue, resolve, reject);                 return;            }        }        state = fulfilled;        value = newValue;        execute();    }     function reject( reason) {        state = rejected;        value = reason;        execute();    }     function execute( ) {        setTimeout( function ( ) {            callbacks.forEach( function ( callback) {                handle(callback);            });        }, 0);    }    fn(resolve, reject); }

上述代码增加了新的 reject方法,供异步操作失败时调用,同时抽出了 resolve和 reject共用的部分,形成 execute方法

错误冒泡是上述代码已经支持,且非常实用的一个特性。在 handle中发现没有指定异步操作失败的回调时,会直接将 bridge promise( then函数返回的 promise,后同)设为 rejected状态,如此达成执行后续失败回调的效果。这有利于简化串行Promise的失败处理成本,因为一组异步操作往往会对应一个实际功能,失败处理方法通常是一致的

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
<span class="line" style="box-sizing: border-box;">
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">//例6</span></span>
<br>
<span class="line" style="box-sizing: border-box;">getUserId()</span>
<br>
<span class="line" style="box-sizing: border-box;">    .then(getUserJobById)</span>
<br>
<span class="line" style="box-sizing: border-box;">    .then(
<span class="function" style="box-sizing: border-box; color: rgb(82, 82, 82);">
<span class="keyword" style="box-sizing: border-box; color: rgb(0, 146, 219);">function</span> (
<span class="params" style="box-sizing: border-box;">job</span>) </span>{</span>
<br>
<span class="line" style="box-sizing: border-box;">        
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">// 处理job</span></span>
<br>
<span class="line" style="box-sizing: border-box;">    },
<span class="function" style="box-sizing: border-box; color: rgb(82, 82, 82);">
<span class="keyword" style="box-sizing: border-box; color: rgb(0, 146, 219);">function</span> (
<span class="params" style="box-sizing: border-box;">error</span>) </span>{</span>
<br>
<span class="line" style="box-sizing: border-box;">        
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">// getUserId或者getUerJobById时出现的错误</span></span>
<br>
<span class="line" style="box-sizing: border-box;">        
<span class="built_in" style="box-sizing: border-box; color: rgb(26, 188, 156);">console</span>.log(error);</span>
<br>
<span class="line" style="box-sizing: border-box;">    });</span>
<br>

2.8 异常处理

如果在执行成功回调、失败回调时代码出错怎么办?对于这类异常,可以使用 try-catch捕获错误,并将 bridge promise设为 rejected状态。 handle方法改造如下

function handle( callback) {     if (state === pending) {        callbacks.push(callback);         return;    }     var cb = state === fulfilled ? callback.onFulfilled : callback.onRejected,        ret;     if (cb === null) {        cb = state === fulfilled ? callback.resolve : callback.reject;        cb(value);         return;    }     try {        ret = cb(value);        callback.resolve(ret);    } catch (e) {        callback.reject(e);    } }

如果在异步操作中,多次执行 resolve或者 reject会重复处理后续回调,可以通过内置一个标志位解决

2.9 完整实现

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
<span class="line" style="box-sizing: border-box;">
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">// 三种状态</span></span>
<br>
<span class="line" style="box-sizing: border-box;">
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">const</span> PENDING =
<span class="string" style="box-sizing: border-box; color: rgb(26, 188, 156);">"pending"</span>;</span>
<br>
<span class="line" style="box-sizing: border-box;">
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">const</span> RESOLVED =
<span class="string" style="box-sizing: border-box; color: rgb(26, 188, 156);">"resolved"</span>;</span>
<br>
<span class="line" style="box-sizing: border-box;">
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">const</span> REJECTED =
<span class="string" style="box-sizing: border-box; color: rgb(26, 188, 156);">"rejected"</span>;</span>
<br>
<span class="line" style="box-sizing: border-box;">
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">// promise 接收一个函数参数,该函数会立即执行</span></span>
<br>
<span class="line" style="box-sizing: border-box;">
<span class="function" style="box-sizing: border-box; color: rgb(82, 82, 82);">
<span class="keyword" style="box-sizing: border-box; color: rgb(0, 146, 219);">function</span>
<span class="title" style="box-sizing: border-box;">MyPromise</span>(
<span class="params" style="box-sizing: border-box;">fn</span>) </span>{</span>
<br>
<span class="line" style="box-sizing: border-box;">  
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">let</span> _this =
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">this</span>;</span>
<br>
<span class="line" style="box-sizing: border-box;">  _this.currentState = PENDING;</span>
<br>
<span class="line" style="box-sizing: border-box;">  _this.value =
<span class="literal" style="box-sizing: border-box; color: rgb(174, 129, 255);">undefined</span>;</span>
<br>
<span class="line" style="box-sizing: border-box;">  
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">// 用于保存 then 中的回调,只有当 promise</span></span>
<br>
<span class="line" style="box-sizing: border-box;">  
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">// 状态为 pending 时才会缓存,并且每个实例至多缓存一个</span></span>
<br>
<span class="line" style="box-sizing: border-box;">  _this.resolvedCallbacks = [];</span>
<br>
<span class="line" style="box-sizing: border-box;">  _this.rejectedCallbacks = [];</span>
<br>
<span class="line" style="box-sizing: border-box;"></span>
<br>
<span class="line" style="box-sizing: border-box;">  _this.resolve =
<span class="function" style="box-sizing: border-box; color: rgb(82, 82, 82);">
<span class="keyword" style="box-sizing: border-box; color: rgb(0, 146, 219);">function</span> (
<span class="params" style="box-sizing: border-box;">value</span>) </span>{</span>
<br>
<span class="line" style="box-sizing: border-box;">    
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">if</span> (value
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">instanceof</span> MyPromise) {</span>
<br>
<span class="line" style="box-sizing: border-box;">      
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">// 如果 value 是个 Promise,递归执行</span></span>
<br>
<span class="line" style="box-sizing: border-box;">      
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">return</span> value.then(_this.resolve, _this.reject)</span>
<br>
<span class="line" style="box-sizing: border-box;">    }</span>
<br>
<span class="line" style="box-sizing: border-box;">    setTimeout(
<span class="function" style="box-sizing: border-box; color: rgb(82, 82, 82);">
<span class="params" style="box-sizing: border-box;">()</span> =></span> {
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">// 异步执行,保证执行顺序</span></span>
<br>
<span class="line" style="box-sizing: border-box;">      
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">if</span> (_this.currentState === PENDING) {</span>
<br>
<span class="line" style="box-sizing: border-box;">        _this.currentState = RESOLVED;</span>
<br>
<span class="line" style="box-sizing: border-box;">        _this.value = value;</span>
<br>
<span class="line" style="box-sizing: border-box;">        _this.resolvedCallbacks.forEach(
<span class="function" style="box-sizing: border-box; color: rgb(82, 82, 82);">
<span class="params" style="box-sizing: border-box;">cb</span> =></span> cb());</span>
<br>
<span class="line" style="box-sizing: border-box;">      }</span>
<br>
<span class="line" style="box-sizing: border-box;">    })</span>
<br>
<span class="line" style="box-sizing: border-box;">  };</span>
<br>
<span class="line" style="box-sizing: border-box;"></span>
<br>
<span class="line" style="box-sizing: border-box;">  _this.reject =
<span class="function" style="box-sizing: border-box; color: rgb(82, 82, 82);">
<span class="keyword" style="box-sizing: border-box; color: rgb(0, 146, 219);">function</span> (
<span class="params" style="box-sizing: border-box;">reason</span>) </span>{</span>
<br>
<span class="line" style="box-sizing: border-box;">    setTimeout(
<span class="function" style="box-sizing: border-box; color: rgb(82, 82, 82);">
<span class="params" style="box-sizing: border-box;">()</span> =></span> {
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">// 异步执行,保证执行顺序</span></span>
<br>
<span class="line" style="box-sizing: border-box;">      
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">if</span> (_this.currentState === PENDING) {</span>
<br>
<span class="line" style="box-sizing: border-box;">        _this.currentState = REJECTED;</span>
<br>
<span class="line" style="box-sizing: border-box;">        _this.value = reason;</span>
<br>
<span class="line" style="box-sizing: border-box;">        _this.rejectedCallbacks.forEach(
<span class="function" style="box-sizing: border-box; color: rgb(82, 82, 82);">
<span class="params" style="box-sizing: border-box;">cb</span> =></span> cb());</span>
<br>
<span class="line" style="box-sizing: border-box;">      }</span>
<br>
<span class="line" style="box-sizing: border-box;">    })</span>
<br>
<span class="line" style="box-sizing: border-box;">  }</span>
<br>
<span class="line" style="box-sizing: border-box;">  
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">// 用于解决以下问题</span></span>
<br>
<span class="line" style="box-sizing: border-box;">  
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">// new Promise(() => throw Error(error))</span></span>
<br>
<span class="line" style="box-sizing: border-box;">  
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">try</span> {</span>
<br>
<span class="line" style="box-sizing: border-box;">    fn(_this.resolve, _this.reject);</span>
<br>
<span class="line" style="box-sizing: border-box;">  }
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">catch</span> (e) {</span>
<br>
<span class="line" style="box-sizing: border-box;">    _this.reject(e);</span>
<br>
<span class="line" style="box-sizing: border-box;">  }</span>
<br>
<span class="line" style="box-sizing: border-box;">}</span>
<br>
<span class="line" style="box-sizing: border-box;"></span>
<br>
<span class="line" style="box-sizing: border-box;">MyPromise.prototype.then =
<span class="function" style="box-sizing: border-box; color: rgb(82, 82, 82);">
<span class="keyword" style="box-sizing: border-box; color: rgb(0, 146, 219);">function</span> (
<span class="params" style="box-sizing: border-box;">onResolved, onRejected</span>) </span>{</span>
<br>
<span class="line" style="box-sizing: border-box;">  
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">var</span> self =
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">this</span>;</span>
<br>
<span class="line" style="box-sizing: border-box;">  
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">// 规范 2.2.7,then 必须返回一个新的 promise</span></span>
<br>
<span class="line" style="box-sizing: border-box;">  
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">var</span> promise2;</span>
<br>
<span class="line" style="box-sizing: border-box;">  
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">// 规范 2.2.onResolved 和 onRejected 都为可选参数</span></span>
<br>
<span class="line" style="box-sizing: border-box;">  
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">// 如果类型不是函数需要忽略,同时也实现了透传</span></span>
<br>
<span class="line" style="box-sizing: border-box;">  
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">// Promise.resolve(4).then().then((value) => console.log(value))</span></span>
<br>
<span class="line" style="box-sizing: border-box;">  
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">typeof</span>
<span class="string" style="box-sizing: border-box; color: rgb(26, 188, 156);">function</span> ? onResolved :
<span class="function" style="box-sizing: border-box; color: rgb(82, 82, 82);">
<span class="params" style="box-sizing: border-box;">v</span> =></span> v;</span>
<br>
<span class="line" style="box-sizing: border-box;">  
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">typeof</span>
<span class="string" style="box-sizing: border-box; color: rgb(26, 188, 156);">function</span> ? onRejected :
<span class="function" style="box-sizing: border-box; color: rgb(82, 82, 82);">
<span class="params" style="box-sizing: border-box;">r</span> =></span>
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">throw</span> r;</span>
<br>
<span class="line" style="box-sizing: border-box;"></span>
<br>
<span class="line" style="box-sizing: border-box;">  
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">if</span> (self.currentState === RESOLVED) {</span>
<br>
<span class="line" style="box-sizing: border-box;">    
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">return</span> (promise2 =
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">new</span> MyPromise(
<span class="function" style="box-sizing: border-box; color: rgb(82, 82, 82);">
<span class="keyword" style="box-sizing: border-box; color: rgb(0, 146, 219);">function</span> (
<span class="params" style="box-sizing: border-box;">resolve, reject</span>) </span>{</span>
<br>
<span class="line" style="box-sizing: border-box;">      
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">// 规范 2.2.4,保证 onFulfilled,onRjected 异步执行</span></span>
<br>
<span class="line" style="box-sizing: border-box;">      
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">// 所以用了 setTimeout 包裹下</span></span>
<br>
<span class="line" style="box-sizing: border-box;">      setTimeout(
<span class="function" style="box-sizing: border-box; color: rgb(82, 82, 82);">
<span class="keyword" style="box-sizing: border-box; color: rgb(0, 146, 219);">function</span> (
<span class="params" style="box-sizing: border-box;"></span>) </span>{</span>
<br>
<span class="line" style="box-sizing: border-box;">        
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">try</span> {</span>
<br>
<span class="line" style="box-sizing: border-box;">          
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">var</span> x = onResolved(self.value);</span>
<br>
<span class="line" style="box-sizing: border-box;">          resolutionProcedure(promise2, x, resolve, reject);</span>
<br>
<span class="line" style="box-sizing: border-box;">        }
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">catch</span> (reason) {</span>
<br>
<span class="line" style="box-sizing: border-box;">          reject(reason);</span>
<br>
<span class="line" style="box-sizing: border-box;">        }</span>
<br>
<span class="line" style="box-sizing: border-box;">      });</span>
<br>
<span class="line" style="box-sizing: border-box;">    }));</span>
<br>
<span class="line" style="box-sizing: border-box;">  }</span>
<br>
<span class="line" style="box-sizing: border-box;"></span>
<br>
<span class="line" style="box-sizing: border-box;">  
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">if</span> (self.currentState === REJECTED) {</span>
<br>
<span class="line" style="box-sizing: border-box;">    
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">return</span> (promise2 =
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">new</span> MyPromise(
<span class="function" style="box-sizing: border-box; color: rgb(82, 82, 82);">
<span class="keyword" style="box-sizing: border-box; color: rgb(0, 146, 219);">function</span> (
<span class="params" style="box-sizing: border-box;">resolve, reject</span>) </span>{</span>
<br>
<span class="line" style="box-sizing: border-box;">      setTimeout(
<span class="function" style="box-sizing: border-box; color: rgb(82, 82, 82);">
<span class="keyword" style="box-sizing: border-box; color: rgb(0, 146, 219);">function</span> (
<span class="params" style="box-sizing: border-box;"></span>) </span>{</span>
<br>
<span class="line" style="box-sizing: border-box;">        
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">// 异步执行onRejected</span></span>
<br>
<span class="line" style="box-sizing: border-box;">        
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">try</span> {</span>
<br>
<span class="line" style="box-sizing: border-box;">          
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">var</span> x = onRejected(self.value);</span>
<br>
<span class="line" style="box-sizing: border-box;">          resolutionProcedure(promise2, x, resolve, reject);</span>
<br>
<span class="line" style="box-sizing: border-box;">        }
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">catch</span> (reason) {</span>
<br>
<span class="line" style="box-sizing: border-box;">          reject(reason);</span>
<br>
<span class="line" style="box-sizing: border-box;">        }</span>
<br>
<span class="line" style="box-sizing: border-box;">      });</span>
<br>
<span class="line" style="box-sizing: border-box;">    }));</span>
<br>
<span class="line" style="box-sizing: border-box;">  }</span>
<br>
<span class="line" style="box-sizing: border-box;"></span>
<br>
<span class="line" style="box-sizing: border-box;">  
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">if</span> (self.currentState === PENDING) {</span>
<br>
<span class="line" style="box-sizing: border-box;">    
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">return</span> (promise2 =
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">new</span> MyPromise(
<span class="function" style="box-sizing: border-box; color: rgb(82, 82, 82);">
<span class="keyword" style="box-sizing: border-box; color: rgb(0, 146, 219);">function</span> (
<span class="params" style="box-sizing: border-box;">resolve, reject</span>) </span>{</span>
<br>
<span class="line" style="box-sizing: border-box;">      self.resolvedCallbacks.push(
<span class="function" style="box-sizing: border-box; color: rgb(82, 82, 82);">
<span class="keyword" style="box-sizing: border-box; color: rgb(0, 146, 219);">function</span> (
<span class="params" style="box-sizing: border-box;"></span>) </span>{</span>
<br>
<span class="line" style="box-sizing: border-box;">        
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">// 考虑到可能会有报错,所以使用 try/catch 包裹</span></span>
<br>
<span class="line" style="box-sizing: border-box;">        
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">try</span> {</span>
<br>
<span class="line" style="box-sizing: border-box;">          
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">var</span> x = onResolved(self.value);</span>
<br>
<span class="line" style="box-sizing: border-box;">          resolutionProcedure(promise2, x, resolve, reject);</span>
<br>
<span class="line" style="box-sizing: border-box;">        }
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">catch</span> (r) {</span>
<br>
<span class="line" style="box-sizing: border-box;">          reject(r);</span>
<br>
<span class="line" style="box-sizing: border-box;">        }</span>
<br>
<span class="line" style="box-sizing: border-box;">      });</span>
<br>
<span class="line" style="box-sizing: border-box;"></span>
<br>
<span class="line" style="box-sizing: border-box;">      self.rejectedCallbacks.push(
<span class="function" style="box-sizing: border-box; color: rgb(82, 82, 82);">
<span class="keyword" style="box-sizing: border-box; color: rgb(0, 146, 219);">function</span> (
<span class="params" style="box-sizing: border-box;"></span>) </span>{</span>
<br>
<span class="line" style="box-sizing: border-box;">        
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">try</span> {</span>
<br>
<span class="line" style="box-sizing: border-box;">          
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">var</span> x = onRejected(self.value);</span>
<br>
<span class="line" style="box-sizing: border-box;">          resolutionProcedure(promise2, x, resolve, reject);</span>
<br>
<span class="line" style="box-sizing: border-box;">        }
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">catch</span> (r) {</span>
<br>
<span class="line" style="box-sizing: border-box;">          reject(r);</span>
<br>
<span class="line" style="box-sizing: border-box;">        }</span>
<br>
<span class="line" style="box-sizing: border-box;">      });</span>
<br>
<span class="line" style="box-sizing: border-box;">    }));</span>
<br>
<span class="line" style="box-sizing: border-box;">  }</span>
<br>
<span class="line" style="box-sizing: border-box;">};</span>
<br>
<span class="line" style="box-sizing: border-box;">
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">// 规范 2.3</span></span>
<br>
<span class="line" style="box-sizing: border-box;">
<span class="function" style="box-sizing: border-box; color: rgb(82, 82, 82);">
<span class="keyword" style="box-sizing: border-box; color: rgb(0, 146, 219);">function</span>
<span class="title" style="box-sizing: border-box;">resolutionProcedure</span>(
<span class="params" style="box-sizing: border-box;">promise2, x, resolve, reject</span>) </span>{</span>
<br>
<span class="line" style="box-sizing: border-box;">  
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">// 规范 2.3.1,x 不能和 promise2 相同,避免循环引用</span></span>
<br>
<span class="line" style="box-sizing: border-box;">  
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">if</span> (promise2 === x) {</span>
<br>
<span class="line" style="box-sizing: border-box;">    
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">return</span> reject(
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">new</span>
<span class="built_in" style="box-sizing: border-box; color: rgb(26, 188, 156);">TypeError</span>(
<span class="string" style="box-sizing: border-box; color: rgb(26, 188, 156);">"Error"</span>));</span>
<br>
<span class="line" style="box-sizing: border-box;">  }</span>
<br>
<span class="line" style="box-sizing: border-box;">  
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">// 规范 2.3.2</span></span>
<br>
<span class="line" style="box-sizing: border-box;">  
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">// 如果 x 为 Promise,状态为 pending 需要继续等待否则执行</span></span>
<br>
<span class="line" style="box-sizing: border-box;">  
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">if</span> (x
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">instanceof</span> MyPromise) {</span>
<br>
<span class="line" style="box-sizing: border-box;">    
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">if</span> (x.currentState === PENDING) {</span>
<br>
<span class="line" style="box-sizing: border-box;">      x.then(
<span class="function" style="box-sizing: border-box; color: rgb(82, 82, 82);">
<span class="keyword" style="box-sizing: border-box; color: rgb(0, 146, 219);">function</span> (
<span class="params" style="box-sizing: border-box;">value</span>) </span>{</span>
<br>
<span class="line" style="box-sizing: border-box;">        
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">// 再次调用该函数是为了确认 x resolve 的</span></span>
<br>
<span class="line" style="box-sizing: border-box;">        
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">// 参数是什么类型,如果是基本类型就再次 resolve</span></span>
<br>
<span class="line" style="box-sizing: border-box;">        
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">// 把值传给下个 then</span></span>
<br>
<span class="line" style="box-sizing: border-box;">        resolutionProcedure(promise2, value, resolve, reject);</span>
<br>
<span class="line" style="box-sizing: border-box;">      }, reject);</span>
<br>
<span class="line" style="box-sizing: border-box;">    }
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">else</span> {</span>
<br>
<span class="line" style="box-sizing: border-box;">      x.then(resolve, reject);</span>
<br>
<span class="line" style="box-sizing: border-box;">    }</span>
<br>
<span class="line" style="box-sizing: border-box;">    
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">return</span>;</span>
<br>
<span class="line" style="box-sizing: border-box;">  }</span>
<br>
<span class="line" style="box-sizing: border-box;">  
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">// 规范 2.3.3.3.3</span></span>
<br>
<span class="line" style="box-sizing: border-box;">  
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">// reject 或者 resolve 其中一个执行过得话,忽略其他的</span></span>
<br>
<span class="line" style="box-sizing: border-box;">  
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">let</span> called =
<span class="literal" style="box-sizing: border-box; color: rgb(174, 129, 255);">false</span>;</span>
<br>
<span class="line" style="box-sizing: border-box;">  
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">// 规范 2.3.3,判断 x 是否为对象或者函数</span></span>
<br>
<span class="line" style="box-sizing: border-box;">  
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">if</span> (x !==
<span class="literal" style="box-sizing: border-box; color: rgb(174, 129, 255);">null</span> && (
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">typeof</span> x ===
<span class="string" style="box-sizing: border-box; color: rgb(26, 188, 156);">"object"</span> ||
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">typeof</span> x ===
<span class="string" style="box-sizing: border-box; color: rgb(26, 188, 156);">"function"</span>)) {</span>
<br>
<span class="line" style="box-sizing: border-box;">    
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">// 规范 2.3.3.2,如果不能取出 then,就 reject</span></span>
<br>
<span class="line" style="box-sizing: border-box;">    
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">try</span> {</span>
<br>
<span class="line" style="box-sizing: border-box;">      
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">// 规范 2.3.3.1</span></span>
<br>
<span class="line" style="box-sizing: border-box;">      
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">let</span> then = x.then;</span>
<br>
<span class="line" style="box-sizing: border-box;">      
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">// 如果 then 是函数,调用 x.then</span></span>
<br>
<span class="line" style="box-sizing: border-box;">      
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">if</span> (
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">typeof</span> then ===
<span class="string" style="box-sizing: border-box; color: rgb(26, 188, 156);">"function"</span>) {</span>
<br>
<span class="line" style="box-sizing: border-box;">        
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">// 规范 2.3.3.3</span></span>
<br>
<span class="line" style="box-sizing: border-box;">        then.call(</span>
<br>
<span class="line" style="box-sizing: border-box;">          x,</span>
<br>
<span class="line" style="box-sizing: border-box;">          y => {</span>
<br>
<span class="line" style="box-sizing: border-box;">            
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">if</span> (called)
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">return</span>;</span>
<br>
<span class="line" style="box-sizing: border-box;">            called =
<span class="literal" style="box-sizing: border-box; color: rgb(174, 129, 255);">true</span>;</span>
<br>
<span class="line" style="box-sizing: border-box;">            
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">// 规范 2.3.3.3.1</span></span>
<br>
<span class="line" style="box-sizing: border-box;">            resolutionProcedure(promise2, y, resolve, reject);</span>
<br>
<span class="line" style="box-sizing: border-box;">          },</span>
<br>
<span class="line" style="box-sizing: border-box;">          e => {</span>
<br>
<span class="line" style="box-sizing: border-box;">            
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">if</span> (called)
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">return</span>;</span>
<br>
<span class="line" style="box-sizing: border-box;">            called =
<span class="literal" style="box-sizing: border-box; color: rgb(174, 129, 255);">true</span>;</span>
<br>
<span class="line" style="box-sizing: border-box;">            reject(e);</span>
<br>
<span class="line" style="box-sizing: border-box;">          }</span>
<br>
<span class="line" style="box-sizing: border-box;">        );</span>
<br>
<span class="line" style="box-sizing: border-box;">      }
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">else</span> {</span>
<br>
<span class="line" style="box-sizing: border-box;">        
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">// 规范 2.3.3.4</span></span>
<br>
<span class="line" style="box-sizing: border-box;">        resolve(x);</span>
<br>
<span class="line" style="box-sizing: border-box;">      }</span>
<br>
<span class="line" style="box-sizing: border-box;">    }
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">catch</span> (e) {</span>
<br>
<span class="line" style="box-sizing: border-box;">      
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">if</span> (called)
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">return</span>;</span>
<br>
<span class="line" style="box-sizing: border-box;">      called =
<span class="literal" style="box-sizing: border-box; color: rgb(174, 129, 255);">true</span>;</span>
<br>
<span class="line" style="box-sizing: border-box;">      reject(e);</span>
<br>
<span class="line" style="box-sizing: border-box;">    }</span>
<br>
<span class="line" style="box-sizing: border-box;">  }
<span class="keyword" style="box-sizing: border-box; color: rgb(233, 105, 0);">else</span> {</span>
<br>
<span class="line" style="box-sizing: border-box;">    
<span class="comment" style="box-sizing: border-box; color: rgb(179, 179, 179);">// 规范 2.3.4,x 为基本类型</span></span>
<br>
<span class="line" style="box-sizing: border-box;">    resolve(x);</span>
<br>
<span class="line" style="box-sizing: border-box;">  }</span>
<br>
<span class="line" style="box-sizing: border-box;">}</span>
<br>

2.10 小结

这里一定要注意的点是: promise里面的 then函数仅仅是注册了后续需要执行的代码,真正的执行是在 resolve方法里面执行的,理清了这层,再来分析源码会省力的多

现在回顾下 Promise的实现过程,其主要使用了设计模式中的观察者模式

通过 Promise.prototype.then和 Promise.prototype.catch方法将观察者方法注册到被观察者 Promise对象中,同时返回一个新的 Promise对象,以便可以链式调用 被观察者管理内部 pending、 fulfilled和 rejected的状态转变,同时通过构造函数中传递的 resolve和 reject方法以主动触发状态转变和通知观察者

三、参考

ruanyifeng-Promise 对象

二维码

扫一扫,关注我们

声明:本文由【益华网络】编辑上传发布,转载此文章须经作者同意,并请附上出处【益华网络】及本页链接。如内容、图片有任何版权问题,请联系我们进行处理。

感兴趣吗?

欢迎联系我们,我们愿意为您解答任何有关网站疑难问题!

您身边的【网站建设专家】

搜索千万次不如咨询1次

主营项目:网站建设,手机网站,响应式网站,SEO优化,小程序开发,公众号系统,软件开发等

立即咨询 15368564009
在线客服
嘿,我来帮您!