Promise簡(jiǎn)單實(shí)現(xiàn)(三)

  1. Promise對(duì)象有以下幾種狀態(tài):
  • pending: 初始狀態(tài), 既不是 fulfilled 也不是 rejected.
  • fulfilled: 成功的操作.
  • rejected: 失敗的操作.
1).Promise的初步框架
function Promise(executor) {
    let self = this; //緩存this卵惦,當(dāng)前promise的實(shí)例
    self.status = 'pending'; //初始狀態(tài)
    self.value = 'undefined';//默認(rèn)成功的值
    self.reason = 'undefined';//默認(rèn)失敗的原因

    function resolve(value) { //成功狀態(tài)
        if (self.status === 'pending') {
            self.status = 'resolved';
            self.value = value;
        }
    }

    function reject(reason) { //失敗狀態(tài)
        if (self.status === 'pending') {
            self.status = 'rejected';
            self.reason = reason;
        }
    }

try{
//因?yàn)榇撕瘮?shù)執(zhí)行可能會(huì)異常贡羔,所以需要捕獲交煞,如果出錯(cuò)了,需要用錯(cuò)誤對(duì)象reject
    executor(resolve, reject)
}catch(e){
    reject(e); // 捕獲的時(shí)候發(fā)生異常,就直接失敗了
  }
}

Promise.prototype.then = function (onFulfilled, onRjected) {
    let self = this;
    if (self.status === 'resloved') {
        onFulfilled(self.value)
    }
    if (self.status === 'rejected') {
        onRjected(self.reason)
    }
}
2).promise實(shí)例可以多次then,當(dāng)成功后會(huì)將then中的成功方法按順序執(zhí)行咖刃,我們可以先將then中的成功的回調(diào)和失敗的回調(diào)存到數(shù)組內(nèi)吟策,當(dāng)成功時(shí)調(diào)用成功的數(shù)組即可棠绘。
function Promise(executor) {
    let self = this;
    self.status = 'pending';
    self.value = 'undefined';
    self.reason = 'undefined';

------------------------------------------------------------------------------
//此處為增加的代碼
    self.onResolvedCallbacks = []; // 存放then成功的回調(diào)
    self.onRejectedCallbacks = []; // 存放then失敗的回調(diào)
------------------------------------------------------------------------------

    function resolve(value) { //成功狀態(tài)
        if (self.status === 'pending') {
            self.status = 'resolved';
            self.value = value;

------------------------------------------------------------------------------
//此處為增加的代碼
            self.onResolvedCallbacks.forEach(function (fn) {
                    fn();
            });
------------------------------------------------------------------------------

        }
    }

    function reject(reason) { //失敗狀態(tài)
        if (self.status === 'pending') {
            self.status = 'rejected';
            self.reason = reason;

------------------------------------------------------------------------------
//此處為增加的代碼
            self.onRejectedCallbacks.forEach(function (fn) {
                    fn();
            });
------------------------------------------------------------------------------

        }
    }

    try{
        executor(resolve, reject)
    }catch(e){
        reject(e)
    }
}

Promise.prototype.then = function (onFulfilled, onRjected) {
    let self = this;
    if (self.status === 'resloved') {
        onFulfilled(self.value)
    }

    if (self.status === 'rejected') {
        onRjected(self.reason)
    }

------------------------------------------------------------------------------
//此處為增加的代碼
    if(self.status === 'pending'){
        self.onResolvedCallbacks.push(function(){
            onFulfilled(self.value)
        })
        self.onRejectedCallbacks.push(function(){
            onRjected(self.reason)
        })
    }
------------------------------------------------------------------------------
}
3).鏈?zhǔn)秸{(diào)用 jquery,jquery能實(shí)現(xiàn)鏈?zhǔn)秸{(diào)用靠的就是返回this,promise不能返回this,promise實(shí)現(xiàn)鏈?zhǔn)秸{(diào)用靠的是返回一個(gè)新的promise
function Promise(executor) {
    let self = this;
    self.status = 'pending';
    self.value = 'undefined';
    self.reason = 'undefined';
    self.onResolvedCallbacks = []; // 存放then成功的回調(diào)
    self.onRejectedCallbacks = []; // 存放then失敗的回調(diào)

    function resolve(value) { //成功狀態(tài)
        if (self.status === 'pending') {
            self.status = 'resolved';
            self.value = value;
            self.onResolvedCallbacks.forEach(function (fn) {
                fn();
            });
        }
    }

    function reject(reason) { //失敗狀態(tài)
        if (self.status === 'pending') {
            self.status = 'rejected';
            self.reason = reason;
            self.onRejectedCallbacks.forEach(function (fn) {
                fn();
            })
        }
    }
    try {
        executor(resolve, reject)
    } catch (e) {
        reject(e)
    }

}

Promise.prototype.then = function (onFulfilled, onRjected) {
------------------------------------------------------------------------------
    //我們的代碼可以在then中什么都不傳promise中值的穿透(實(shí)際使用)
    //如果成功和失敗默認(rèn)不傳翠语,給一個(gè)函數(shù)(規(guī)范規(guī)定)
    onFulfilled = typeof onFulfilled === 'function' ? onFulfilled : function (value) {
        return value
    };
    onRjected = typeof onRjected === 'function' ? onRjected : function (err) {
            throw err;
        }
------------------------------------------------------------------------------

        let self = this;
------------------------------------------------------------------------------
//增加的代碼
    let promise2; //返回的promise
------------------------------------------------------------------------------

    if (self.status === 'resolved') {
------------------------------------------------------------------------------
//增加的代碼
        promise2 = new Promise(function (resolve, reject) {
            // 當(dāng)成功或者失敗執(zhí)行時(shí)有異常那么返回的promise應(yīng)該處于失敗狀態(tài)
            // x可能是一個(gè)promise 也有可能是一個(gè)普通的值
            try {
                let x = onFulfilled(self.value);
                // x可能是別人promise叽躯,寫一個(gè)方法統(tǒng)一處理
                resolvePromise(promise2, x, resolve, reject);
            } catch (e) {
                reject(e);
            }
        })
------------------------------------------------------------------------------
    }
    if (self.status === 'rejected') {
------------------------------------------------------------------------------
//增加的代碼  
        promise2 = new Promise(function (resolve, reject) {
            try {
                let x = onRjected(self.reason);
                resolvePromise(promise2, x, resolve, reject);
            } catch (e) {
                reject(e);
            }
        })
------------------------------------------------------------------------------
        
    }
    if (self.status === 'pending') {
        self.onResolvedCallbacks.push(function () {
------------------------------------------------------------------------------
//增加的代碼            
            try {
                let x = onFulfilled(self.value);
                resolvePromise(promise2, x, resolve, reject);
            } catch (e) {
                reject(e)
            }
        })
        self.onRejectedCallbacks.push(function () {
            try {
                let x = onRjected(self.reason);
                resolvePromise(promise2, x, resolve, reject);
            } catch (e) {
                reject(e);
            }
------------------------------------------------------------------------------
        })
    }
------------------------------------------------------------------------------
//增加的代碼
    return promise2;
------------------------------------------------------------------------------
}
4).解析then中的結(jié)果
function resolvePromise(promise2, x, resolve, reject) {
    // 有可能這里返回的x是別人的promise
    // 盡可能允許其他亂寫
    //返回的結(jié)果和promise是同一個(gè)那么永遠(yuǎn)不會(huì)成功和失敗
    if (promise2 === x) { //這里應(yīng)該報(bào)一個(gè)類型錯(cuò)誤,有問題
        return reject(new TypeError('循環(huán)引用了'))
    }
    // 看x是不是一個(gè)promise,promise應(yīng)該是一個(gè)對(duì)象
   //有些人寫的promise可能會(huì)既調(diào)用成功 又調(diào)用失敗肌括,如果兩個(gè)都調(diào)用先 調(diào)用誰另一個(gè)就忽略掉
    let called; // 表示是否調(diào)用過成功或者失敗
  
    if (x !== null && (typeof x === 'object' || typeof x === 'function')) {
        // 可能是promise {},看這個(gè)對(duì)象中是否有then方法险毁,如果有then我就認(rèn)為他是promise了
        try { // {then:1}
            let then = x.then;
            if (typeof then === 'function') {
                // 成功
                then.call(x, function (y) {
                    if (called) return
                    called = true
                    // y可能還是一個(gè)promise,在去解析直到返回的是一個(gè)普通值
                    resolvePromise(promise2, y, resolve, reject)
                }, function (err) { //失敗
                    if (called) return
                    called = true
                    reject(err);
                })
            } else {
                resolve(x)
            }
        } catch (e) {
            if (called) return
            called = true;
            reject(e);
        }
    } else { // 說明是一個(gè)普通值1
        resolve(x); // 表示成功了
    }
}
如果then中無論是成功的回調(diào)還是失敗的回調(diào)只要返回了結(jié)果就會(huì)走下一個(gè)then中的成功们童,如果有錯(cuò)誤走下一個(gè)then的失敗
如果第一個(gè)promise返回一個(gè)普通值,會(huì)進(jìn)到下一次then的成功的回調(diào),如果第一個(gè)promise返回了一個(gè)promise,需要等待返回的promise執(zhí)行后的結(jié)果傳遞給下一次then中
6).捕獲錯(cuò)誤的方法
Promise.prototype.catch = function (callback) {
    return this.then(null, callback)
}
7).Promise的靜態(tài)方法
  • all:全部成功才算成功
Promise.all = function (promises) {
    //promises是一個(gè)promise的數(shù)組
    return new Promise(function (resolve, reject) {
        let arr = []; //arr是最終返回值的結(jié)果
        let i = 0; // 表示成功了多少次
        function processData(index, y) {
            arr[index] = y;
            if (++i === promises.length) {
                resolve(arr);
            }
        }
        for (let i = 0; i < promises.length; i++) {
            promises[i].then(function (y) {
                processData(i, y)
            }, reject)
        }
    })
}
  • race:只要有一個(gè)promise成功了 就算成功鲸鹦。如果第一個(gè)失敗了就失敗了
Promise.race = function (promises) {
    return new Promise(function (resolve, reject) {
        for (var i = 0; i < promises.length; i++) {
            promises[i].then(resolve,reject)
        }
    })
}
  • resolve:生成一個(gè)成功的promise
Promise.resolve = function(value){
    return new Promise(function(resolve,reject){
        resolve(value);
    })
}
  • reject:生成一個(gè)失敗的promise
Promise.reject = function(reason){
    return new Promise(function(resolve,reject){
        reject(reason);
    })
}
8).promise規(guī)范中要求慧库,所有的onFufiled和onRjected都需要異步執(zhí)行,setTimeout(在onFufiled和onRjected處添加setTimeout)
Promise.prototype.then = function (onFulfilled, onRjected) {
    //成功和失敗默認(rèn)不穿給一個(gè)函數(shù)
    onFulfilled = typeof onFulfilled === 'function' ? onFulfilled : function (value) {
        return value;
    }
    onRjected = typeof onRjected === 'function' ? onRjected : function (err) {
        throw err;
    }
    let self = this;
    let promise2; //返回的promise
    if (self.status === 'resolved') {
        promise2 = new Promise(function (resolve, reject) {
            // 當(dāng)成功或者失敗執(zhí)行時(shí)有異常那么返回的promise應(yīng)該處于失敗狀態(tài)
            // x可能是一個(gè)promise 也有可能是一個(gè)普通的值
-------------------------------------------------------------------------------------------------
            setTimeout(function () {
                try {
                    let x = onFulfilled(self.value);
                    // x可能是別人promise,寫一個(gè)方法統(tǒng)一處理
                    resolvePromise(promise2, x, resolve, reject);
                } catch (e) {
                    reject(e);
                }
            })
        })
    }
    if (self.status === 'rejected') {
        promise2 = new Promise(function (resolve, reject) {
            setTimeout(function () {
                try {
                    let x = onRjected(self.reason);
                    resolvePromise(promise2, x, resolve, reject);
                } catch (e) {
                    reject(e);
                }
            })

        })
    }
    // 當(dāng)調(diào)用then時(shí)可能沒成功 也沒失敗
    if (self.status === 'pending') {
        promise2 = new Promise(function (resolve, reject) {
            // 此時(shí)沒有resolve 也沒有reject
            self.onResolvedCallbacks.push(function () {
                setTimeout(function () {
                    try {
                        let x = onFulfilled(self.value);
                        resolvePromise(promise2, x, resolve, reject);
                    } catch (e) {
                        reject(e)
                    }
                })
            });
            self.onRejectedCallbacks.push(function () {
                setTimeout(function () {
                    try {
                        let x = onRjected(self.reason);
                        resolvePromise(promise2, x, resolve, reject);
                    } catch (e) {
                        reject(e);
                    }
                })
            });
        })
    }
-------------------------------------------------------------------------------------------------
    return promise2;
}
9).promise測(cè)試

下載一個(gè)Promise的測(cè)試庫,promises-aplus-tests

  • npm install -g promises-aplus-tests
  • promises-aplus-tests 文件名
測(cè)試需要添加的代碼
Promise.defer = Promise.deferred = function () {
    let dfd = {};
    dfd.promise = new Promise(function (resolve, reject) {
        dfd.resolve = resolve;
        dfd.reject = reject;
    });
    return dfd
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子炫加,更是在濱河造成了極大的恐慌碳褒,老刑警劉巖,帶你破解...
    沈念sama閱讀 210,978評(píng)論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件叠纷,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)济舆,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 89,954評(píng)論 2 384
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來莺债,“玉大人滋觉,你說我怎么就攤上這事∑氚睿” “怎么了椎侠?”我有些...
    開封第一講書人閱讀 156,623評(píng)論 0 345
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)措拇。 經(jīng)常有香客問我我纪,道長(zhǎng),這世上最難降的妖魔是什么丐吓? 我笑而不...
    開封第一講書人閱讀 56,324評(píng)論 1 282
  • 正文 為了忘掉前任浅悉,我火速辦了婚禮,結(jié)果婚禮上汰蜘,老公的妹妹穿的比我還像新娘仇冯。我一直安慰自己,他們只是感情好族操,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,390評(píng)論 5 384
  • 文/花漫 我一把揭開白布苛坚。 她就那樣靜靜地躺著比被,像睡著了一般。 火紅的嫁衣襯著肌膚如雪泼舱。 梳的紋絲不亂的頭發(fā)上等缀,一...
    開封第一講書人閱讀 49,741評(píng)論 1 289
  • 那天,我揣著相機(jī)與錄音娇昙,去河邊找鬼尺迂。 笑死,一個(gè)胖子當(dāng)著我的面吹牛冒掌,可吹牛的內(nèi)容都是我干的噪裕。 我是一名探鬼主播,決...
    沈念sama閱讀 38,892評(píng)論 3 405
  • 文/蒼蘭香墨 我猛地睜開眼股毫,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼膳音!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起铃诬,我...
    開封第一講書人閱讀 37,655評(píng)論 0 266
  • 序言:老撾萬榮一對(duì)情侶失蹤祭陷,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后趣席,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體兵志,經(jīng)...
    沈念sama閱讀 44,104評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,451評(píng)論 2 325
  • 正文 我和宋清朗相戀三年宣肚,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了想罕。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,569評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡钉寝,死狀恐怖弧呐,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情嵌纲,我是刑警寧澤俘枫,帶...
    沈念sama閱讀 34,254評(píng)論 4 328
  • 正文 年R本政府宣布,位于F島的核電站逮走,受9級(jí)特大地震影響鸠蚪,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜师溅,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,834評(píng)論 3 312
  • 文/蒙蒙 一茅信、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧墓臭,春花似錦蘸鲸、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,725評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽膝舅。三九已至,卻和暖如春窑多,著一層夾襖步出監(jiān)牢的瞬間仍稀,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,950評(píng)論 1 264
  • 我被黑心中介騙來泰國打工埂息, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留技潘,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 46,260評(píng)論 2 360
  • 正文 我出身青樓千康,卻偏偏與公主長(zhǎng)得像享幽,于是被迫代替她去往敵國和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子拾弃,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,446評(píng)論 2 348

推薦閱讀更多精彩內(nèi)容

  • Promise 對(duì)象 Promise 的含義 Promise 是異步編程的一種解決方案琉闪,比傳統(tǒng)的解決方案——回調(diào)函...
    neromous閱讀 8,700評(píng)論 1 56
  • 目錄:Promise 的含義基本用法Promise.prototype.then()Promise.prototy...
    BluesCurry閱讀 1,490評(píng)論 0 8
  • 00、前言Promise 是異步編程的一種解決方案斯入,比傳統(tǒng)的解決方案——回調(diào)函數(shù)和事件——更合理和更強(qiáng)大砂碉。它由社區(qū)...
    夜幕小草閱讀 2,128評(píng)論 0 12
  • Promiese 簡(jiǎn)單說就是一個(gè)容器,里面保存著某個(gè)未來才會(huì)結(jié)束的事件(通常是一個(gè)異步操作)的結(jié)果刻两,語法上說增蹭,Pr...
    雨飛飛雨閱讀 3,352評(píng)論 0 19
  • 項(xiàng)目結(jié)構(gòu)圖如下: 下面分步驟一一講解:第一步,在pom.xml中添加相應(yīng)依賴: <?xml version="1....
    simple_xuan閱讀 515評(píng)論 0 0