本文首发于 vivo互联网技术 微信公众号
连接: https://mp.weixin.qq.com/s/UNzYgpnKzmW6bAapYxnXRQ
做者:孔垂亮es6
不少同窗在学习 Promise 时,知其然殊不知其因此然,对其中的用法理解不了。本系列文章由浅入深逐步实现 Promise,并结合流程图、实例以及动画进行演示,达到深入理解 Promise 用法的目的。数组
本系列文章有以下几个章节组成:promise
图解 Promise 实现原理(一)—— 基础实现bash
图解 Promise 实现原理(二)—— Promise 链式调用微信
图解 Promise 实现原理(三)—— Promise 原型方法实现异步
图解 Promise 实现原理(四)—— Promise 静态方法实现函数
本文适合对 Promise 的用法有所了解的人阅读,若是还不清楚,请自行查阅阮一峰老师的 《ES6入门 之 Promise 对象》。学习
Promise 规范有不少,如 Promise/A,Promise/B,Promise/D 以及 Promise/A 的升级版 Promise/A+,有兴趣的能够去了解下,最终 ES6 中采用了 Promise/A+ 规范。因此本文的Promise源码是按照Promise/A+规范来编写的(不想看英文版的移步Promise/A+规范中文翻译)。动画
为了让你们更容易理解,咱们从一个场景开始,一步一步跟着思路思考,会更容易看懂。ui
考虑下面一种获取用户 id 的请求处理:
//不使用Promise
http.get('some_url', function (result) {
//do something
console.log(result.id);
});
//使用Promise
new Promise(function (resolve) {
//异步请求
http.get('some_url', function (result) {
resolve(result.id)
})
}).then(function (id) {
//do something
console.log(id);
})复制代码
//不使用Promise
http.get('some_url', function (id) {
//do something
http.get('getNameById', id, function (name) {
//do something
http.get('getCourseByName', name, function (course) {
//dong something
http.get('getCourseDetailByCourse', function (courseDetail) {
//do something
})
})
})
});
//使用Promise
function getUserId(url) {
return new Promise(function (resolve) {
//异步请求
http.get(url, function (id) {
resolve(id)
})
})
}
getUserId('some_url').then(function (id) {
//do something
return getNameById(id); // getNameById 是和 getUserId 同样的Promise封装。下同
}).then(function (name) {
//do something
return getCourseByName(name);
}).then(function (course) {
//do something
return getCourseDetailByCourse(course);
}).then(function (courseDetail) {
//do something
});复制代码
说到底,Promise 也仍是使用回调函数,只不过是把回调封装在了内部,使用上一直经过 then 方法的链式调用,使得多层的回调嵌套看起来变成了同一层的,书写上以及理解上会更直观和简洁一些。
//极简的实现
class Promise {
callbacks = [];
constructor(fn) {
fn(this._resolve.bind(this));
}
then(onFulfilled) {
this.callbacks.push(onFulfilled);
}
_resolve(value) {
this.callbacks.forEach(fn => fn(value));
}
}
//Promise应用
let p = new Promise(resolve => {
setTimeout(() => {
console.log('done');
resolve('5秒');
}, 5000);
}).then((tip) => {
console.log(tip);
})复制代码
调用 then 方法,将想要在 Promise 异步操做成功时执行的 onFulfilled 放入callbacks队列,其实也就是注册回调函数,能够向观察者模式方向思考;
建立 Promise 实例时传入的函数会被赋予一个函数类型的参数,即 resolve,它接收一个参数 value,表明异步操做返回的结果,当异步操做执行成功后,会调用resolve方法,这时候其实真正执行的操做是将 callbacks 队列中的回调一一执行。
(图:基础版本实现原理)
首先 new Promise 时,传给 Promise 的函数设置定时器模拟异步的场景,接着调用 Promise 对象的 then 方法注册异步操做完成后的 onFulfilled,最后当异步操做完成时,调用 resolve(value), 执行 then 方法注册的 onFulfilled。
then 方法注册的 onFulfilled 是存在一个数组中,可见 then 方法能够调用屡次,注册的多个onFulfilled 会在异步操做完成后根据添加的顺序依次执行。以下:
//then 的说明
let p = new Promise(resolve => {
setTimeout(() => {
console.log('done');
resolve('5秒');
}, 5000);
});
p.then(tip => {
console.log('then1', tip);
});
p.then(tip => {
console.log('then2', tip);
});复制代码
//极简的实现+链式调用
class Promise {
callbacks = [];
constructor(fn) {
fn(this._resolve.bind(this));
}
then(onFulfilled) {
this.callbacks.push(onFulfilled);
return this;//看这里
}
_resolve(value) {
this.callbacks.forEach(fn => fn(value));
}
}
let p = new Promise(resolve => {
setTimeout(() => {
console.log('done');
resolve('5秒');
}, 5000);
}).then(tip => {
console.log('then1', tip);
}).then(tip => {
console.log('then2', tip);
});复制代码
(图:基础版本的链式调用)
上面 Promise 的实现存在一个问题:若是在 then 方法注册 onFulfilled 以前,resolve 就执行了,onFulfilled 就不会执行到了。好比上面的例子中咱们把 setTimout 去掉:
//同步执行了resolve
let p = new Promise(resolve => {
console.log('同步执行');
resolve('同步执行');
}).then(tip => {
console.log('then1', tip);
}).then(tip => {
console.log('then2', tip);
});复制代码
这显然是不容许的,Promises/A+规范明确要求回调须要经过异步方式执行,用以保证一致可靠的执行顺序。所以要加入一些处理,保证在 resolve 执行以前,then 方法已经注册完全部的回调:
//极简的实现+链式调用+延迟机制
class Promise {
callbacks = [];
constructor(fn) {
fn(this._resolve.bind(this));
}
then(onFulfilled) {
this.callbacks.push(onFulfilled);
return this;
}
_resolve(value) {
setTimeout(() => {//看这里
this.callbacks.forEach(fn => fn(value));
});
}
}复制代码
(图:延迟机制)
可是这样依然存在问题,在 resolve 执行后,再经过 then 注册上来的 onFulfilled 都没有机会执行了。以下所示,咱们加了延迟后,then1 和 then2 能够打印出来了,但下例中的 then3 依然打印不出来。因此咱们须要增长状态,而且保存 resolve 的值。
let p = new Promise(resolve => {
console.log('同步执行');
resolve('同步执行');
}).then(tip => {
console.log('then1', tip);
}).then(tip => {
console.log('then2', tip);
});
setTimeout(() => {
p.then(tip => {
console.log('then3', tip);
})
});复制代码
为了解决上一节抛出的问题,咱们必须加入状态机制,也就是你们熟知的 pending、fulfilled、rejected。
Promises/A+ 规范中明确规定了,pending 能够转化为 fulfilled 或 rejected 而且只能转化一次,也就是说若是 pending 转化到 fulfilled 状态,那么就不能再转化到 rejected。而且 fulfilled 和 rejected 状态只能由 pending 转化而来,二者之间不能互相转换。
增长状态后的实现是这样的
//极简的实现+链式调用+延迟机制+状态
class Promise {
callbacks = [];
state = 'pending';//增长状态
value = null;//保存结果
constructor(fn) {
fn(this._resolve.bind(this));
}
then(onFulfilled) {
if (this.state === 'pending') {//在resolve以前,跟以前逻辑同样,添加到callbacks中
this.callbacks.push(onFulfilled);
} else {//在resolve以后,直接执行回调,返回结果了
onFulfilled(this.value);
}
return this;
}
_resolve(value) {
this.state = 'fulfilled';//改变状态
this.value = value;//保存结果
this.callbacks.forEach(fn => fn(value));
}
}复制代码
注意:当增长完状态以后,原先的_resolve中的定时器能够去掉了。当reolve同步执行时,虽然callbacks为空,回调函数尚未注册上来,但没有关系,由于后面注册上来时,判断状态为fulfilled,会当即执行回调。
(图:Promise 状态管理)
实现源码中只增长了 fulfilled 的状态 和 onFulfilled 的回调,但为了完整性,在示意图中增长了 rejected 和 onRejected 。后面章节会实现。
resolve 执行时,会将状态设置为 fulfilled ,并把 value 的值存起来,在此以后调用 then 添加的新回调,都会当即执行,直接返回保存的value值。
(Promise 状态变化演示动画)
详情请点击: https://mp.weixin.qq.com/s/UNzYgpnKzmW6bAapYxnXRQ
至此,一个初具功能的Promise就实现好了,它实现了 then,实现了链式调用,实现了状态管理等等。但仔细想一想,链式调用的实现只是在 then 中 return 了 this,由于是同一个实例,调用再屡次 then 也只能返回相同的一个结果,这显然是不能知足咱们的要求的。下一节,讲述如何实现真正的链式调用。
更多内容敬请关注 vivo 互联网技术 微信公众号
注:转载文章请先与微信号:labs2020 联系。