手写promise
const PENDING = 'pending'
const RESOLVED = 'resolved'
const REJECTED = 'rejected'
function MyPromise(fn) {
// 保存初始化状态
var self = this
// 初始化状态
this.state = PENDING
// 用于保存 resolve 或者 rejected 传入的值
this.value = null
// 用于保存 resolve 的回调函数
this.resolvedCallbacks = []
// 用于保存 reject 的回调函数
this.rejectedCallbacks = []
// 状态转变为 resolved 方法
function resolve(value) {
// 判断传入元素是否为 Promise 值,若是是,则状态改变必须等待前一个状态改变后再进行改变
if (value instanceof MyPromise) {
return value.then(resolve, reject)
}
// 保证代码的执行顺序为本轮事件循环的末尾
setTimeout(() => {
// 只有状态为 pending 时才能转变,
if (self.state === PENDING) {
// 修改状态
self.state = RESOLVED
// 设置传入的值
self.value = value
// 执行回调函数
self.resolvedCallbacks.forEach((callback) => {
callback(value)
})
}
}, 0)
}
// 状态转变为 rejected 方法
function reject(value) {
// 保证代码的执行顺序为本轮事件循环的末尾
setTimeout(() => {
// 只有状态为 pending 时才能转变
if (self.state === PENDING) {
// 修改状态
self.state = REJECTED
// 设置传入的值
self.value = value
// 执行回调函数
self.rejectedCallbacks.forEach((callback) => {
callback(value)
})
}
}, 0)
}
// 将两个方法传入函数执行
try {
fn(resolve, reject)
} catch (e) {
// 遇到错误时,捕获错误,执行 reject 函数
reject(e)
}
}
MyPromise.prototype.then = function (onResolved, onRejected) {
// 首先判断两个参数是否为函数类型,由于这两个参数是可选参数
onResolved =
typeof onResolved === 'function'
? onResolved
: function (value) {
return value
}
onRejected =
typeof onRejected === 'function'
? onRejected
: function (error) {
throw error
}
// 若是是等待状态,则将函数加入对应列表中
if (this.state === PENDING) {
this.resolvedCallbacks.push(onResolved)
this.rejectedCallbacks.push(onRejected)
}
// 若是状态已经凝固,则直接执行对应状态的函数
if (this.state === RESOLVED) {
onResolved(this.value)
}
if (this.state === REJECTED) {
onRejected(this.value)
}
}
手写Promise.all()
Promise.all = function(iterator) {
if (!Array.isArray(iterator)) return;
let count = 0;
let res = [];
return new Promise((resolve, reject) => {
for(let item of iterator) {
Promise.resolve(item)
.then(data => {
res[count++] = data;
if (count === iterator.length) {
resolve(res);
}
})
.catch(e => {
reject(e);
})
}
})
}
// test
let p1 = Promise.resolve(3);
let p2 = 4;
let p3 = new Promise(resolve => {
setTimeout(resolve, 100, 'lee')
// setTimeout的第三个日后参数都是用来做为第一个参数也就是函数的参数,也就是实际上是setTimeout(resolve('lee'), 100)
});
Promise.all([p1, p2, p3]).then(data => {
console.log(data);
})
手写Promise.race()
Promise.race = function(iterator) {
return new Promise((resolve, reject) => {
for(let item of iterator) {
Promise.resolve(item)
.then(data => {
resolve(data)
})
.catch(e => {
reject(e)
})
}
})
}
let p1 = new Promise(resolve => {
setTimeout(resolve, 105, 'p1 done')
})
let p2 = new Promise(resolve => {
setTimeout(resolve, 100, 'p2 done')
})
Promise.race([p1, p2]).then(data => {
console.log(data); // p2 done
})