Promise源码解密-then的链式调用

简介: Promise源码解密-then的链式调用

系列文章

  1. Promise源码解密-PromisesA+标准
  2. Promise源码解密-同步版
  3. Promise源码解密-异步版
  4. Promise源码解密-then的链式调用
  5. Promise源码解密-catch/resolve/reject/race

创作不易 拒绝白嫖 点个赞呗

关注专栏 Promise源码解密,带你走进Promise的深处!!!


then的用法

  1. 通过return 给下一个then传值,成功/失败的返回值,最后都是传到下一个then的成功会回调
let t = new Promise((resolve, reject) => {
    console.log(a)  // 执行console会报错,最终只会执行reject
})
t.then((res)=>{
    return res
},error=>{
    return error
}).then(
  (res) => {
    console.log(res.toString(),'res===='); // ReferenceError: a is not defined res====
  },
  (error) => {
    console.log(error);
  }
);
  1. then中的onFulfilled/onReject 是可选的, 下面的例子,第一个then没有return值,但是第二个then也可以接受到值,最终输出 then 回调可选。
let t = new Promise((resolve, reject) => {
    resolve('then 回调可选')
})
t.then().then(
  (res) => {
    console.log(res);
  },
  (error) => {
    console.log(error);
  }
);
  1. 实例不可 循环引用,报错 : Chaining cycle detected for promise #
let p = new Promise((resolve, reject) => {
    resolve('不可循环引用')
})
var p2=p.then(()=>{
   return p2 // 返回的值还是p2
})

完整代码

// 定义状态常量
const STATUS_PENDING = "pending";
const STATUS_FULFILLED = "fulfilled";
const STATUS_REJECTED = "rejected";
function resolvePromise(promise2, x, resolve, reject) {
    // 用来保存是否已经reject或者resolve过
    let called
    if (promise2 === x) {
        throw new TypeError('Chaining cycle detected for promise')
    }
    // 如果是函数或者object的话先默认是promise
    if (x != null && (typeof x === 'object' || typeof x === 'function')) {
        try {
            let then = x.then
            // 如果then是函数的话
            if (typeof then === 'function') {
                // 为啥不直接x.then()
                // 因为then已经判断过是不是function,但是x.then没有判断过
                // 就让then执行 第一个参数是this   后面是成功的回调 和 失败的回调
                //  这里的y是啥,如果x是promsie的话,那么y就是x中的resolve/reject的值
                then.call(x, y => {
                    // 成功和失败只能调用一个
                    if (called) return;
                    called = true;
                    console.log(y, 'yyyyyyyyyyyy')
                    // resolve的结果依旧是promise 那就继续解析
                    resolvePromise(promise2, y, resolve, reject);
                }, err => {
                    // 成功和失败只能调用一个
                    if (called) return;
                    called = true;
                    reject(err);
                })
            } else {
                resolve(x); // 如果不是函数,那就直接返回结果
            }
        } catch (error) {
            // 成功和失败只能调用一个
            if (called) return;
            called = true;
            // 没有then 不是函数也不是普通值
            reject(error)
        }
    } else {
        // x 是一个普通值
        resolve(x)
    }
}
class MyPromise {
    // 接收一个 执行器
    // new Promise的时候,执行器立即执行,
    // 执行的时候根据指定函数执行
    // 并且程序报错的情况下
    constructor(executor) {
        this.status = STATUS_PENDING;
        this.value = undefined;
        this.reason = undefined;
        // 这里的两个数据,相当于订阅者的篮子,then相当于订阅者,接收到发布者的函数,保存在篮子里
        this.onFulfilledCallBacks = [];
        this.onRejectCallBacks = [];
        try {
            // 这里将传入的resolve和reject变为类中的调用的话。
            // (resolve, reject) => {
            //     resolve("then链式调用");=>this.resolve(then链式调用)
            // }
            executor(this.resolve, this.reject);
        } catch (error) {
            //  报错的话直接将错误原因作为reason
            this.reject(error);
        }
    }
    // 这里为啥使用箭头函数而不是resolve(){}
    // 这是因为上面的executor调用的时候,resolve中的this其实是undefined
    // executor单独调用了resolve/reject 这时,这两个方法存在于执行是的环境,this不再指向实例
    // 箭头函数内部的this总是指向定义时所在的对象
    resolve = (value) => {
        // 判断状态是不是
        if (this.status === STATUS_PENDING) {
            console.log("执行resolve");
            // 这里接收传进来的结果
            this.value = value;
            this.status = STATUS_FULFILLED;
            this.onFulfilledCallBacks.forEach((fn) => {
                fn();
            });
        }
    };
    reject = (reason) => {
        if (this.status === STATUS_PENDING) {
            console.log("执行reject");
            // 这里接收传进来的错误的结果
            this.reason = reason;
            this.status = STATUS_REJECTED;
            // 这里相当于发布的动作
            this.onRejectCallBacks.forEach((fn) => {
                fn();
            });
        }
    };
    then(onFulfilled, onReject) {
        onFulfilled =
            typeof onFulfilled === "function" ? onFulfilled : (value) => value;
        onReject = typeof onReject === "function" ? onReject : (reason) => reason;
        console.log("执行then时的状态: ", this.status);
        let promise2 = new MyPromise((resolve, reject) => {
            if (this.status === STATUS_FULFILLED) {
                // 这个地方使用settimeout的目的是获取到promise2。因为setTimeout是异步的,
                // 会等到赋值给promise2结束后才执行,
                // 这里的resolvePromise的作用是为了区分x是普通值还是新的promise还是函数
                // 如果直接resolve(x)的话会导致,输出的是[object Object] res====,并不是想要的值
                setTimeout(() => {
                    // 这里try进行了捕获异常,在最外面的executor不是进行了捕获了?
                    // 这是因为异步里面的方法不会被最外层的那个捕获到
                    try {
                        console.log("执行 onFulfilled");
                        //   这里的x是啥? 是then中回调的return的返回值
                        let x = onFulfilled(this.value);
                        resolvePromise(promise2, x, resolve, reject);
                    } catch (e) {
                        reject(e);
                    }
                }, 0);
            }
            if (this.status === STATUS_REJECTED) {
                setTimeout(() => {
                    try {
                        console.log("执行 onReject");
                        let x = onReject(this.reason);
                        resolvePromise(promise2, x, resolve, reject);
                    } catch (e) {
                        reject(e);
                    }
                }, 0);
            }
            if (this.status === STATUS_PENDING) {
                this.onFulfilledCallBacks.push(() =>
                    setTimeout(() => {
                        try {
                            let x = onFulfilled(this.value);
                            resolvePromise(promise2, x, resolve, reject);
                        } catch (e) {
                            reject(e);
                        }
                    }, 0)
                );
                this.onRejectCallBacks.push(() =>
                    setTimeout(() => {
                        try {
                            let x = onRejected(this.reason);
                            resolvePromise(promise2, x, resolve, reject);
                        } catch (e) {
                            reject(e);
                        }
                    }, 0)
                );
            }
        });
        return promise2;
    }
}
// 下面的函数相当于发布者
let t = new MyPromise((resolve, reject) => {
    // 指定执行哪个,resolve和reject最终只会执行一个
    // console.log(a)  // 执行console会报错,最终只会执行reject
    // // resolve("res")
    // reject('reject')
    // setTimeout(() => {
    resolve("then链式调用");
    // }, 1000)
});
t.then(() => {
    return new MyPromise((resolve) => {
        console.log(2);
        setTimeout(() => {
            resolve(3);
        }, 3000);
    });
}).then(
    (res) => {
        console.log(res.toString(), "res====");
    },
    (error) => {
        console.log(error);
    }
);

相关文章
|
前端开发
7 # promise 的 then 方法
7 # promise 的 then 方法
83 0
|
6月前
|
前端开发
Promise链式调用与错误处理
Promise链式调用是处理异步操作的方式,它按顺序执行多个任务,每个任务返回Promise对象。通过`.then()`指定成功后的操作,`.catch()`处理错误。示例代码展示了如何使用fetch获取数据,根据状态码解析响应并处理数据,错误则通过`.catch()`捕获。另一个例子定义了三个异步函数构成Promise链,依次执行并处理结果,错误同样由`.catch()`统一管理。
|
4月前
|
前端开发 JavaScript
Vue 中 Promise 的then方法异步使用及async/await 异步使用总结
Vue 中 Promise 的then方法异步使用及async/await 异步使用总结
140 1
|
3月前
|
存储 JSON 前端开发
JavaScript异步编程3——Promise的链式使用
JavaScript异步编程3——Promise的链式使用
30 0
|
6月前
|
前端开发 JavaScript
Promise的链式调用案例讲解
Promise的链式调用案例讲解
|
6月前
|
前端开发
Promise的链式调用
Promise的链式调用是指在一个Promise对象上连续调用多个then方法的过程。通过链式调用,可以将多个异步操作按照顺序执行,并且可以在每个操作完成后处理返回的结果。
66 0
|
6月前
|
前端开发
【源码共读】如何优雅的处理 Promise 的错误
【源码共读】如何优雅的处理 Promise 的错误
102 0
|
前端开发 程序员
promise的链式调用和promise的嵌套的实现
promise的链式调用和promise的嵌套的实现
189 0
|
前端开发 JavaScript API
📕重学JavaScript:Promise 的then()、catch() 和 finally()
大部分时候,你要用的 Promise 对象是 Web API 或第三方 API 返回的。我们要设置 Promise 对象,让它在变成 fulfilled 的时候执行我们想要的成功的代码,而在变成 rejected 的时候执行我们想要的失败的代码。
348 0
📕重学JavaScript:Promise 的then()、catch() 和 finally()
|
前端开发 JavaScript
web前端面试高频考点——JavaScript 篇(二)【JS 异步进阶】Event Loop、then 和 catch、async/await、宏任务微任务、手撕 Promise 源码
web前端面试高频考点——JavaScript 篇(二)【JS 异步进阶】Event Loop、then 和 catch、async/await、宏任务微任务、手撕 Promise 源码
174 0