使用JavaScript按部就班实现一个简单的Promise,支持异步和then链式调用。
翻译并整理自 Medium: Implementing a simple Promise in Javascript - by Zhi Sun
在前端面试和平常开发中,常常会接触到Promise。而且在现现在的不少面试中,也会常常被要求手写Promise。javascript
接下来,将使用JavaScript按部就班实现一个简单的Promise,支持异步和then链式调用。前端
Promise对象用于表示一个异步操做的最终完成 (或失败)及其结果值,经常使用来实现异步操做。java
Promise有三种状态:面试
pending数组
初始状态promise
fulfilled异步
执行成功后的状态函数
rejected测试
执行失败后的状态this
Promise状态只能由pending
改变为fulfilled
或者由pending
改变为rejected
,Promise状态改变的这一过程被称为settled
,而且,状态一旦改变,后续就不会再次被改变。
Promise构造函数接收一个函数参数executor
,该函数接收两个参数:
执行resolve
会将Promise状态由pending
改变为fulfilled
,并触发then
方法中的成功回调函数onFulfilled
,
执行reject
会将Promise状态由pending
改变为rejected
,并触发then
方法中的失败回调函数onRejected
。
then
方法接收两个参数:
onFulfilled
成功回调函数,接收一个参数,即resolve
函数中传入的值
onRejected
失败回调函数,接收一个参数,即reject
函数中传入的值
若是Promise状态变为fulfilled
,就会执行成功回调函数onFulfilled
;若是Promise状态变为rejected
,就会执行失败回调函数onRejected
。
首先,constructor
接收一个函数executor
,该函数又接收两个参数,分别是resolve
和reject
函数。
所以,须要在constructor
中建立resolve
和reject
函数,并传入executor
函数中。
class MyPromise { constructor(executor) { const resolve = (value) => {}; const reject = (reason) => {}; try { executor(resolve, reject); } catch (err) { reject(err); } } }
其次,Promise会根据状态,执行对应的回调函数。最开始的状态为pending
,当resolve
时,状态由pending
变为fulfilled
;当reject
时,状态由pending
变为rejected
。而且,一旦状态变动后,就不会再次变动。
class MyPromise { constructor(executor) { this.state = 'pending'; const resolve = (value) => { if (this.state === 'pending') { this.state = 'fulfilled'; } }; const reject = (reason) => { if (this.state === 'pending') { this.state = 'rejected'; } }; try { executor(resolve, reject); } catch (err) { reject(err); } } }
Promise状态变动后,会触发then
方法中对应的回调函数。若是状态由pending
变为fulfilled
,则会触发成功回调函数,若是状态由pending
变为rejected
,则会触发失败回调函数。
class MyPromise { constructor(executor) { this.state = 'pending'; this.value = null; const resolve = (value) => { if (this.state === 'pending') { this.state = 'fulfilled'; this.value = value; } }; const reject = (reason) => { if (this.state === 'pending') { this.state = 'rejected'; this.value = reason; } }; try { executor(resolve, reject); } catch (err) { reject(err); } } then(onFulfilled, onRejected) { if (this.state === 'fulfilled') { onFulfilled(this.value); } if (this.state === 'rejected') { onRejected(this.value); } } }
接下来能够写点测试代码测试一下功能
const p1 = new MyPromise((resolve, reject) => resolve('resolved')); p1.then( (res) => console.log(res), // resolved (err) => console.log(err) ); const p2 = new MyPromise((resolve, reject) => reject('rejected')); p2.then( (res) => console.log(res), (err) => console.log(err) // rejected );
可是,若是用如下代码测试,会发现什么也没有输出。
const p1 = new MyPromise((resolve, reject) => { setTimeout(() => resolve('resolved'), 1000); }); p1.then( (res) => console.log(res), (err) => console.log(err) ); const p2 = new MyPromise((resolve, reject) => { setTimeout(() => reject('rejected'), 1000); }); p2.then( (res) => console.log(res), (err) => console.log(err) );
这是由于在调用then
方法时,Promise仍处于pending
状态。onFulfilled
和onRejected
回调函数都没有被执行。
所以,接下来须要支持异步。
为了支持异步,须要先保存onFulfilled
和onRejected
回调函数,一旦Promise状态变化,马上执行对应的回调函数。
⚠:这里有个细节须要注意,即onFulfilledCallbacks
和onRejectedCallbacks
是数组,由于Promise可能会被调用屡次,所以会存在多个回调函数。
class MyPromise { constructor(executor) { this.state = 'pending'; this.value = null; this.onFulfilledCallbacks = []; this.onRejectedCallbacks = []; const resolve = (value) => { if (this.state === 'pending') { this.state = 'fulfilled'; this.value = value; this.onFulfilledCallbacks.forEach((fn) => fn(value)); } }; const reject = (value) => { if (this.state === 'pending') { this.state = 'rejected'; this.value = value; this.onRejectedCallbacks.forEach((fn) => fn(value)); } }; try { executor(resolve, reject); } catch (err) { reject(err); } } then(onFulfilled, onRejected) { if (this.state === 'pending') { this.onFulfilledCallbacks.push(onFulfilled); this.onRejectedCallbacks.push(onRejected); } if (this.state === 'fulfilled') { onFulfilled(this.value); } if (this.state === 'rejected') { onRejected(this.value); } } }
接下来能够用以前的测试代码测试一下功能
const p1 = new MyPromise((resolve, reject) => { setTimeout(() => resolve('resolved'), 1000); }); p1.then( (res) => console.log(res), // resolved (err) => console.log(err) ); const p2 = new MyPromise((resolve, reject) => { setTimeout(() => reject('rejected'), 1000); }); p2.then( (res) => console.log(res), (err) => console.log(err) // rejected );
可是若是用如下代码测试,会发现报错了。
const p1 = new MyPromise((resolve, reject) => { setTimeout(() => resolve('resolved'), 1000); }); p1.then( (res) => console.log(res), (err) => console.log(err) ).then( (res) => console.log(res), (err) => console.log(err) ); // Uncaught TypeError: Cannot read property 'then' of undefined
这是由于第一个then
方法并无返回任何值,然而却又连续调用了then
方法。
所以,接下来须要实现then
链式调用。
then
链式调用的Promise要想支持then
链式调用,then
方法须要返回一个新的Promise。
所以,须要改造一下then
方法,返回一个新的Promise,等上一个Promise的onFulfilled
或onRejected
回调函数执行完成后,再执行新的Promise的resolve
或reject
函数。
class MyPromise { then(onFulfilled, onRejected) { return new MyPromise((resolve, reject) => { if (this.state === 'pending') { this.onFulfilledCallbacks.push(() => { try { const fulfilledFromLastPromise = onFulfilled(this.value); resolve(fulfilledFromLastPromise); } catch (err) { reject(err); } }); this.onRejectedCallbacks.push(() => { try { const rejectedFromLastPromise = onRejected(this.value); reject(rejectedFromLastPromise); } catch (err) { reject(err); } }); } if (this.state === 'fulfilled') { try { const fulfilledFromLastPromise = onFulfilled(this.value); resolve(fulfilledFromLastPromise); } catch (err) { reject(err); } } if (this.state === 'rejected') { try { const rejectedFromLastPromise = onRejected(this.value); reject(rejectedFromLastPromise); } catch (err) { reject(err); } } }); } }
接下来能够用如下代码测试一下功能
const p1 = new MyPromise((resolve, reject) => { setTimeout(() => resolve('resolved'), 1000); }); p1.then( (res) => { console.log(res); // resolved return res; }, (err) => console.log(err) ).then( (res) => console.log(res), // resolved (err) => console.log(err) ); const p2 = new MyPromise((resolve, reject) => { setTimeout(() => reject('rejected'), 1000); }); p2.then( (res) => console.log(res), (err) => { console.log(err); // rejected throw new Error('rejected'); } ).then( (res) => console.log(res), (err) => console.log(err) // Error: rejected );
可是,若是改用如下代码测试,会发现第二个then
方法中的成功回调函数并无按预期输出(‘resolved’),而是输出了上一个then
方法的onFulfilled
回调函数中返回的Promise。
const p1 = new MyPromise((resolve, reject) => { setTimeout(() => resolve('resolved'), 1000); }); p1.then( (res) => { console.log(res); // resolved return new MyPromise((resolve, reject) => { setTimeout(() => resolve('resolved'), 1000); }) }, (err) => console.log(err) ).then( (res) => console.log(res), // MyPromise {state: "pending"} (err) => console.log(err) );
这是由于onFulfilled
/onRejected
回调函数执行完以后,只是简单的将onFulfilled
/onRejected
执行完返回的值传入resolve
/reject
函数中执行,并无考虑onFulfilled
/onRejected
执行完会返回一个新的Promise这种状况,因此第二次then
方法的成功回调函数中直接输出了上一次then
方法的成功回调函数中返回的Promise。所以,接下来须要解决这个问题。
首先,能够将以上测试代码改为另外一种写法,方便梳理思路。
const p1 = new MyPromise((resolve, reject) => { setTimeout(() => resolve('resolved'), 1000); }); const p2 = p1.then( (res) => { console.log(res); const p3 = new MyPromise((resolve, reject) => { setTimeout(() => resolve('resolved'), 1000); }); return p3; }, (err) => console.log(err) ); p2.then( (res) => console.log(res), (err) => console.log(err) );
能够看到,一共有三个Promise:
第一个Promise
即经过new
构造出来的p1
第二个Promise
即经过调用then
方法返回的p2
第三个Promise
即在p1.then
方法的成功回调函数参数中返回的p3
如今的问题是,调用p2的then
方法时,p3还处于pending
状态。
要想实现p2.then
方法中的回调函数能正确输出p3中resolve
/reject
以后的值,须要先等p3状态变化后再将变化后的值传入p2中的resolve
/reject
中便可。换句话说,三个Promise状态变化的前后顺序应该是p1 --> p3 --> p2。
class MyPromise { then(onFulfilled, onRejected) { return new MyPromise((resolve, reject) => { if (this.state === 'pending') { this.onFulfilledCallbacks.push(() => { try { const fulfilledFromLastPromise = onFulfilled(this.value); if (fulfilledFromLastPromise instanceof MyPromise) { fulfilledFromLastPromise.then(resolve, reject); } else { resolve(fulfilledFromLastPromise); } } catch (err) { reject(err); } }); this.onRejectedCallbacks.push(() => { try { const rejectedFromLastPromise = onRejected(this.value); if (rejectedFromLastPromise instanceof MyPromise) { rejectedFromLastPromise.then(resolve, reject); } else { reject(rejectedFromLastPromise); } } catch (err) { reject(err); } }); } if (this.state === 'fulfilled') { try { const fulfilledFromLastPromise = onFulfilled(this.value); if (fulfilledFromLastPromise instanceof MyPromise) { fulfilledFromLastPromise.then(resolve, reject); } else { resolve(fulfilledFromLastPromise); } } catch (err) { reject(err); } } if (this.state === 'rejected') { try { const rejectedFromLastPromise = onRejected(this.value); if (rejectedFromLastPromise instanceof MyPromise) { rejectedFromLastPromise.then(resolve, reject); } else { reject(rejectedFromLastPromise); } } catch (err) { reject(err); } } }); } }
最后,一个简单的Promise就完成了,支持异步和then
链式调用。完整代码以下:
class MyPromise { constructor(executor) { this.state = 'pending'; this.value = null; this.onFulfilledCallbacks = []; this.onRejectedCallbacks = []; const resolve = (value) => { if (this.state === 'pending') { this.state = 'fulfilled'; this.value = value; this.onFulfilledCallbacks.forEach((fn) => fn(value)); } }; const reject = (value) => { if (this.state === 'pending') { this.state = 'rejected'; this.value = value; this.onRejectedCallbacks.forEach((fn) => fn(value)); } }; try { executor(resolve, reject); } catch (err) { reject(err); } } then(onFulfilled, onRejected) { return new Promise((resolve, reject) => { if (this.state === 'pending') { this.onFulfilledCallbacks.push(() => { try { const fulfilledFromLastPromise = onFulfilled(this.value); if (fulfilledFromLastPromise instanceof Promise) { fulfilledFromLastPromise.then(resolve, reject); } else { resolve(fulfilledFromLastPromise); } } catch (err) { reject(err); } }); this.onRejectedCallbacks.push(() => { try { const rejectedFromLastPromise = onRejected(this.value); if (rejectedFromLastPromise instanceof Promise) { rejectedFromLastPromise.then(resolve, reject); } else { reject(rejectedFromLastPromise); } } catch (err) { reject(err); } }); } if (this.state === 'fulfilled') { try { const fulfilledFromLastPromise = onFulfilled(this.value); if (fulfilledFromLastPromise instanceof Promise) { fulfilledFromLastPromise.then(resolve, reject); } else { resolve(fulfilledFromLastPromise); } } catch (err) { reject(err); } } if (this.state === 'rejected') { try { const rejectedFromLastPromise = onRejected(this.value); if (rejectedFromLastPromise instanceof Promise) { rejectedFromLastPromise.then(resolve, reject); } else { reject(rejectedFromLastPromise); } } catch (err) { reject(err); } } }); } }