myPromsie
- 是个构造函数
- 传入一个可执行函数 函数的入参第一个为 fullFill函数 第二个为 reject函数; 函数立即执行, 参数函数异步执行
- 状态一旦更改就不可以变更 只能 pending => fulfilled 或者 pending => rejected
- then 的时候要处理入参的情况 successCallback 和failCallback 均可能为非函数,默认的 failCallback 一定要将异常抛出, 这样下一个promise便可将其捕获 异常冒泡的目的
- then 中执行回调的时候要捕获异常 将其传给下一个promise如果promise状态未变更 则将回调方法添加到对应队列中,如果promise状态已经变更 需要异步处理成功或者失败回调,因为可能出现 回调结果和当前then返回的Promise一致 从而导致死循环问题
- catch只是then的一种特殊的写法 方便理解和使用
- finally 特点 1. 不过resolve或者reject都会执行 2. 回调没有参数 3. 返回一个Promise 且值可以穿透到下一个then或者catch
- Promise.resolve, Promise.reject 根据其参数返回对应的值 或者状态的Promise即可
- Proise.all 特点1. 返回一个Promise2. 入参是数组 resolve的情况下出参也是数组 且结果顺序和调用顺序一致 3. 所有的值或者promise都完成才能resolve 所有要计数 4. 只要有一个为reject 返回的Promise便reject
- Proise.race 特点 1. 返回一个Promise 2. 入参是数组 那么出参根据第一个成功或者失败的参数来确定 3. 只要有一个resolve 或者reject 便更改返回Promise的状态
const PENDING = "pending";
const FULFILLED = "fulfilled";
const REJECTED = "rejected";
function asyncExecFun(fn) {
setTimeout(() => fn(), 0);
}
function resolvePromise(promise, res, resolve, reject) {
if (promise === res) {
reject(new TypeError("Chaining cycle detected for promise #<MyPromise>"));
return;
}
if (res instanceof MyPromise) {
res.then(resolve, reject);
} else {
resolve(res);
}
}
class MyPromise {
status = PENDING;
value = undefined;
reason = undefined;
successCallbacks = [];
failCallbacks = [];
constructor(exector) {
try {
exector(
(value) => asyncExecFun(() => this.resolve(value)),
(reason) => asyncExecFun(() => this.reject(reason))
);
} catch (e) {
this.reject(e)
}
}
resolve(value) {
if (this.status !== PENDING) return;
this.value = value;
this.status = FULFILLED;
while (this.successCallbacks.length) this.successCallbacks.shift()();
}
reject(reason) {
if (this.status !== PENDING) return;
this.reason = reason;
this.status = REJECTED;
if(!this.failCallbacks.length){
throw '(in MyPromise)'
}
while (this.failCallbacks.length) this.failCallbacks.shift()();
}
then(successCallback, failCallback) {
successCallback =
typeof successCallback == "function" ? successCallback : (v) => v;
failCallback =
typeof failCallback == "function"
? failCallback
: (reason) => {
throw reason;
};
let promise = new MyPromise((resolve, reject) => {
const execFun = (fn, val) => {
try {
let res = fn(val);
resolvePromise(promise, res, resolve, reject);
} catch (e) {
reject(e);
}
};
const execSuccessCallback = () => execFun(successCallback, this.value);
const execFailCallback = () => execFun(failCallback, this.reason);
if (this.status === PENDING) {
this.successCallbacks.push(execSuccessCallback);
this.failCallbacks.push(execFailCallback);
return;
}
asyncExecFun(() => {
if (this.status === FULFILLED) {
execSuccessCallback();
} else if (this.status === REJECTED) {
execFailCallback();
}
});
});
return promise;
}
catch(failCallback) {
return this.then(undefined, failCallback);
}
finally(callback) {
return this.then(
(value) => MyPromise.resolve(callback()).then(() => value),
(reason) =>
MyPromise.resolve(callback()).then(() => {
throw reason;
})
);
}
static resolve(value) {
if (value instanceof MyPromise) return value;
return new MyPromise((resolve) => resolve(value));
}
static reject(reason) {
if (reason instanceof MyPromise) return reason;
return new MyPromise((resolve, reject) => reject(reason));
}
static all(array) {
let result = [];
let len = array.length;
let promise = new MyPromise((resolve, reject) => {
let index = 0;
function addData(key, data) {
result[key] = data;
index++;
if (index == len) {
resolve(result);
}
}
for (let i = 0; i < len; i++) {
let curr = array[i];
if (curr instanceof MyPromise) {
curr.then((value) => addData(i, value), reject);
} else {
addData(i, curr);
}
}
});
return promise;
}
static race(array) {
let promise = new MyPromise((resolve, reject) => {
for (let i = 0; i < array.length; i++) {
let curr = array[i];
if (curr instanceof MyPromise) {
curr.then(resolve, reject);
} else {
resolve(curr);
}
}
});
return promise;
}
}
function resolvePromise(promise2,x,resolve,reject) {
if(promise2===x){
return reject (new TypeError ('Chaining cycle detected for promise #<Promise>'))
}
if(x instanceof myPromise){
x.then(resolve,reject)
}else{
resolve(x)
}
}