Promise是个什么鬼?实现一个Promise.

1. 为何产生promise?

  • 相信大家在写JS的时候写过回调函数,当写了多层回调的时候,逻辑已经很难理清楚了,这就是promise解决的问题: 回调地狱
  • 多层嵌套的回调中,有同步/异步的方法,那么执行顺序会混乱,而promise采用链式调用。

2. 什么是promise?

  • 所谓promise, 翻译成中文为“承诺, 诺言”, 比如, 你承诺这个月挣钱了给你老婆买一个包, 那么你先去挣钱, 等挣钱了就立刻给老婆买包,实现你的诺言,没挣到钱就立马道歉。换成代码就是:

          // 先许下承诺
          const geilaopomaibao = new Promise((resolve, reject) => {
              // 开始挣钱了
              if (money) {
                  // 挣到了
                  reslove('10k');
              } else {
                  // 没挣着钱
                  reject('因为没发工资,这个月没挣到钱');
              }
          })
          
          // 不管挣没挣到钱,要给老婆一个交代。
          geilaopomaibao.then(
              res => {
                  // res就是挣到的钱
                  maibao(res);
              },
              reason => {
                  // 道歉
                  // reason 就是理由
              }
          );
    
  • promise是一种规范,也就是大家都可以实现自己的promise,只要遵循规范,都是可以的。在ES6中也正式引入了promise,当然还有很多库去实现了promise规范,列如: Q,bluebird
    规范:
    Promises/A+
    Promises/A+ 中文

3. 如何使用promise?

- 首先来个简单的列子: <br/>

 ```
    const task = new Promise((resolve, reject) => {
        setTimeout(() => {
            resolve('成功');
        }, 3000);
    })
    
    task.then((res) => {
        // 3秒后打印成功
        console.log(res);
    })
 ```
 看这个列子似乎没什么特别之处,感觉就是把callback换成resolve, 那么再看看下面一个
 
 ```
    // 获取列表之前,需要先判断是否登录,then的链式调用
    new Promise((resolve, reject) => {
        $.ajax('/api/user/login', {
            success: function(isLogin) {
                if (isLogin) {
                    resolve(data);
                } else {
                    reject('没有登录');
                }
            }
        })
    }).then(
        res => {
            $.ajax('/api/list', {
                success: function(data) {
                    resolve(data);
                }
            })
        },
        reason => {
            alert(reason)
        }
    ).then((data) => {
        // data为list数据
        // 在这里还可以为list排序
        return data.sort();
    }).then((list) => {
        // 好了,这里的拿到list 可以使用了
    })
 ```
 - promise有几个静态方法,无需实例化既可使用;
   1. Promise.all() (传入一个promise数组, 等待所有代码完成, 或第一个代码失败)
   
      ```
        const task1 = {ji: 2018};
        
        const task2 = new Promise((resolve, reject) => {
            setTimeout(() => {
                 resolve('2018年');
            }, 5000);
        });
        
        const task3 = new Promise((resolve, reject) => {
            setTimeout(() => {
                 resolve('新年好');
            }, 2000);
        }); 
        
        Promise.all([task1, task2, task3]).then(values => { 
            console.log(values); 
            // 5秒后打印 [{ji: 2018}, '2018年', '新年好'] 
        });
      ```
   2. Promise.race() (传入一个promise数组,当其中一个完成或失败,则采用该promise值)
    
     ```
        const task2 = new Promise((resolve, reject) => {
            setTimeout(() => {
                 resolve('2018年');
            }, 5000);
        }); ;
        const task3 = new Promise((resolve, reject) => {
            setTimeout(() => {
                 resolve('新年好');
            }, 2000);
        }); 
        
        Promise.race([task1, task2, task3]).then(values => { 
            console.log(values); 
            // 2秒后打印 '新年好'
        });
     ```
   3. Promise.resolve() (传入一个值,返回一个同步执行成功状态的promise, 适用于: 如果有方法要求返回一个promise, 但你已经有一个值了, 则这个时候使用)

   ```
        /**
         * @param {Promise} fn 需要传入一个promise对象
         */
        const task = (fn) => {
            fn.then() 
            // ........
        }
        
        Promise.resolve(1);
   ```
   4. Promise. reject() (传入一个值,返回一个同步执行的失败状态的promise)

4. 实现一个promise

class Promise {
    constructor(task) {

        // 当前状态 存在 { pending fulfilled rejected } 3种状态
        this.status = 'pending';

        // 当resolve时的数据
        this.resolveData = null;

        // 当reject时的数据
        this.rejectData = null;

        // 存放所有回调 实现同步调用
        this.onFulfilledList = [];
        this.onRejectedList = [];

        // 执行传入任务
        try {
            task(this.onResolve.bind(this), this.onReject.bind(this));
        } catch (e) {
            this.onReject(e);
        }
    }

    /**
     * 成功,通过
     * 
     * @param {any} data resolve数据
     */
    onResolve(data) {
        if (this.status === 'pending') {
            this.status = 'fulfilled';
            this.resolveData = data;
            this.onFulfilledList.forEach(fn => {
                fn(this.resolveData);
            })
        }
    }

    /**
     * 拒绝,失败
     * 
     * @param {any} data reject数据
     */
    onReject(data) {
        if (this.status === 'pending') {
            this.status = 'rejected';
            this.rejectData = data;
            this.onRejectedList.forEach(fn => {
                fn(this.rejectData);
            })
        }
    }

    /**
     * 回调
     *
     * @param {function} onFulfilled 成功回调
     * @param {function} onRejected 失败回调
     * @return {*}
     */
    then(onFulfilled, onRejected) {
        if (typeof onFulfilled !== 'function') {
            onFulfilled = () => {};
        }

        if (typeof onRejected !== 'function') {
            onRejected = () => {};
        }

        let promise2;

        switch (this.status) {
            case 'pending':
                promise2 = new Promise((resolve,reject) => {
                    this.onFulfilledList.push(() => {
                        let x = onFulfilled(this.resolveData);
                        this.resolvePromise(promise2, x, resolve, reject);
                    });
                    this.onRejectedList.push(() => {
                        let x = onRejected(this.rejectData);
                        this.resolvePromise(promise2, x, resolve, reject);
                    });
                });
                break;
            case 'fulfilled':
                promise2 = new Promise((resolve, reject) => {
                    let x = onFulfilled(this.resolveData);
                    this.resolvePromise(promise2, x, resolve, reject);
                });
                break;
            case 'rejected':
                promise2 = new Promise((resolve, reject) => {
                    let x = onRejected(this.rejectData);
                    this.resolvePromise(promise2, x, resolve, reject);
                });
                break;
            default:
                throw 'promise status error';
        }

        return promise2;
    }

    /**
     * catch方法 等于 reject
     *
     * @param onRejected
     */
    catch(onRejected) {
        if (typeof onRejected !== 'function') {
            onRejected = () => {};
        }

        let promise2;

        promise2 = new Promise((resolve, reject) => {
            let x = onRejected(this.rejectData);
            this.resolvePromise(promise2, x, resolve, reject);
        });
    }

    /**
     * 递归方法,多次then
     *
     * @param {Promise} promise2 第二个promise
     * @param {any} x 上一次then所返回的数据
     * @param {function} resolve 当前的成功方法
     * @param {function} reject 当前的失败方法
     * @return {*}
     */
    resolvePromise(promise2, x, resolve, reject) {
        // 第二个then方法
        let then;
        // 如果x 等于 promise2 重复调用
        if(promise2 === x){
            return reject(new TypeError('循环引用'));
        }

        if (x instanceof Promise) {
            if(x.status === 'pending'){
                x.then(function(y){
                    resolvePromise(promise2, y, resolve, reject);
                }, reject);
            } else if (x.status === 'fulfilled'){
                resolve(x.resolveData);
            } else if (x.status === 'rejected'){
                reject(x.rejectData);
            }
        } else if (x != null && (typeof x === 'object' || typeof x === 'function')) {
            try {
                // 取出下一次then方法
                then = x.then;
                // 如果then为fn 递归调用then方法
                if (typeof then == 'function') {
                    then.call(x, function (y) {
                        resolvePromise(promise2, y, resolve, reject)
                    }, reject);
                } else {
                    // 如果then 不为fn 则以x为值fulfill promise
                    resolve(x);
                }
            } catch (e) {
                reject(e);
            };
        } else {
            resolve(x);
        }
    }

    /**
     * all方法 传入数组,等待所有promise完成
     * 1. 如果有一个为reject,则返回reject
     *
     * @param {Array} promiseList 需要执行的数组
     * @return {Promise}
     */
    static all(promiseList) {
        // 如果传入不是数组,抛出错误
        if (!Array.isArray(promiseList)) {
            new TypeError('must be an array');
        }
        return new Promise((resolve, reject) => {
            // 如果传入数组长度为0 则直接resolve
            if (promiseList.length === 0) {
                resolve([]);
            }

            const allData = new Array();

            let resI = 0;

            promiseList.forEach((item, index) => {
                if (item instanceof Promise) {
                    item.then(
                        res => {
                            allData[index] = res;
                            resolveAll();
                        },
                        rej => {
                            reject(rej);
                        }
                    )
                } else {
                    allData[index] = item;
                    resolveAll();
                }
            })

            function resolveAll() {
                resI++;

                if (resI === promiseList.length) {
                    resolve(allData);
                }
            }
        });
    }

    /**
     * race 方法 当某一个promise完成,则返回
     *
     * @param {Array} promiseList 需要执行的数组
     * @return {Promise}
     */
    static race(promiseList) {
        // 如果传入不是数组,抛出错误
        if (!Array.isArray(promiseList)) {
            new TypeError('must be an array');
        }

        return new Promise((resolve, reject) => {
            // 如果传入数组长度为0 则直接resolve
            if (promiseList.length === 0) {
                resolve([]);
            }

            promiseList.forEach(item => {
                promiseList.forEach(item => {
                    if (item instanceof Promise) {
                        item.then(
                            res => {
                                resolve(res);
                            },
                            rej => {
                                reject(rej);
                            }
                        )
                    } else {
                        resolve(item);
                    }
                })
            })
        })
    }

    /**
     * resolve方法
     *
     * @param value
     * @return {Promise}
     */
    static resolve(value) {
        return new Promise((resolve, reject) => {
            if (value instanceof Promise) {
                value.then(res => {
                    resolve(res);
                })
            } else {
                resolve(value);
            }
        })
    }

    /**
     * reject方法
     *
     * @param reason
     * @return {Promise}
     */
    static reject(reason) {
        return new Promise((resolve, reject) => {
            reject(reason);
        })
    }
}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 199,902评论 5 468
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 84,037评论 2 377
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 146,978评论 0 332
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,867评论 1 272
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,763评论 5 360
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,104评论 1 277
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,565评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,236评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,379评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,313评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,363评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,034评论 3 315
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,637评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,719评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,952评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,371评论 2 346
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,948评论 2 341

推荐阅读更多精彩内容

  • Promise 对象 Promise 的含义 Promise 是异步编程的一种解决方案,比传统的解决方案——回调函...
    neromous阅读 8,688评论 1 56
  • 你不知道JS:异步 第三章:Promises 在第二章,我们指出了采用回调来表达异步和管理并发时的两种主要不足:缺...
    purple_force阅读 2,037评论 0 4
  • 本文适用的读者 本文写给有一定Promise使用经验的人,如果你还没有使用过Promise,这篇文章可能不适合你,...
    HZ充电大喵阅读 7,288评论 6 19
  • 前言 本文旨在简单讲解一下javascript中的Promise对象的概念,特性与简单的使用方法。并在文末会附上一...
    _暮雨清秋_阅读 2,178评论 0 3
  • 你走了 我的心跟着碎了 即使你回来 也是支离破碎的 你结婚的时候记得通知我 不能和你手牵手迈入殿堂 我也要成为你的...
    夏目的温暖阅读 235评论 0 2