Promise

更新

最近看到一個實現(xiàn)胆胰,講得更加清晰,于是重新實現(xiàn)了一遍

class GePromise {
    constructor(fn) {
        this.state = 'pending'
        this.value = undefined
        this.resolvedCb = []
        this.rejectedCb = []

        let self = this
        try {
            let resolve = self.__resolve.bind(self)
            let reject = self.__reject.bind(self)
            fn(resolve, reject)
        } catch (e) {
            self.__reject(e)
        }
    }
    __resolve(value) {
        let self = this
        if (value instanceof GePromise) {
            return value.then(self.__resolve, self.__reject)
        }
        setTimeout(() => {
            self.state = 'resolved'
            self.value = value
            self.resolvedCb.forEach(cb => cb())
        }, 0)
    }
    __reject(reason) {
        let self = this
        setTimeout(() => {
            self.state = 'rejected'
            self.value = reason
            self.rejectedCb.forEach(cb => cb())
        }, 0)
    }
    then(onResolved, onRejected) {
        let self = this
        let state = this.state
        let newPromise
        if (state === 'resolved') {
            newPromise = new GePromise((resolve, reject) => {
                setTimeout(() => {
                    try {
                        let x = onResolved(self.value)
                        self.value = x
                        resolve(x)
                    } catch (reason) {
                        reject(reason)
                    }
                }, 0)
            })
        } else if (state === 'rejected') {
            newPromise = new GePromise((resolve, reject) => {
                setTimeout(() => {
                    try {
                        let x = onRejected(self.value)
                        self.value = x
                        resolve(x)
                    } catch (reason) {
                        reject(reason)
                    }
                }, 0)
            })
        } else if (state === 'pending') {
            newPromise = new GePromise((resolve, reject) => {
                self.resolvedCb.push(() => {
                    try {
                        let x = onResolved(self.value)
                        self.value = x
                        resolve(x)
                    } catch (reason) {
                        reject(reason)
                    }
                })
                self.rejectedCb.push(() => {
                    try {
                        let x = onRejected(self.value)
                        self.value = x
                        resolve(x)
                    } catch (reason) {
                        reject(reason)
                    }
                })
            })
        }
        return newPromise
    }
}


const simpleTest = () => {
    let a = new GePromise((res, rej) => {
        setTimeout(() => {
            console.log('set timeout')
            res('resolve')
        }, 1000)
    })
    let b = a.then((res) => {
        console.log(res) // resolve
        return 'after resolve'
    }, (rej) => {
        console.error(rej)
        return 'first reject'
    }).then(res => {
        console.log(res) // after resolve
    })
}

原文

之前看到朋友實現(xiàn)了一個promise垮斯,覺得挺好玩的郎仆,心中躍躍欲試,于是近期看了一些文章兜蠕,自己也跟著實現(xiàn)了一遍扰肌,記錄下來。
promise是為了解決回調(diào)地獄熊杨,它是通過then方法來注冊回調(diào)函數(shù)曙旭,使得代碼在組織上更加清晰盗舰。

一個簡單的雛形:

class GePromise {
    constructor(fn) {
        this.value = null
        this.callbacks = []
        this.init(fn)
    }
    init(fn) {
        const self = this
        const resolve = function(value) {
            self.callbacks.forEach((callback) => {
                callback(value)
            })
        }
        fn(resolve)
    }
    then(fn) {
        this.callbacks.push(fn)
        return this
    }
}

const test_1 = function () {
    const p = new GePromise((resolve, reject) => {
        setTimeout(() => {
            resolve(1)
        }, 1000)
        console.log(2)
    })
    p.then((val) => {
        console.log(val)
    })
}
test_1()

這時候會有一個問題,有可能在then注冊好回調(diào)之前桂躏,resolve就執(zhí)行了钻趋,因此加入延時機制:

...
const resolve = function(value) {
    setTimeout(() => {
        self.callbacks.forEach((callback) => {
            callback(value)
        })
    }, 0)
}
...

引入狀態(tài)控制:

class GePromise {
    constructor(fn) {
        this.state = 'pending'
        this.value = null
        this.callbacks = []
        this._init(fn)
    }
    _init(fn) {
        const resolve = this._resolve.bind(this)
        fn(resolve)
    }
    _resolve(value) {
        const self = this
        self.value = value
        self.state = 'fulfilled'
        setTimeout(() => {
            self.callbacks.forEach((callback) => {
                callback(value)
            })
        }, 0)
    }
    then(fn) {
        if (this.state === 'pending') {
            this.callbacks.push(fn)
        } else {
            const val = this.value
            fn(val)
        }
        return this
    }
}

const test = function () {
    const p = new GePromise((resolve, reject) => {
        setTimeout(() => {
            resolve(1)
        }, 1000)
        console.log(2)
    })
    p.then((val) => {
        console.log(val)
    })
    setTimeout(() => {
        p.then((val) => {
            console.log(3)
        })
    }, 3000)
}
test()

最后加入reject:

class GePromise {
    constructor(fn) {
        this.state = 'pending'
        this.value = null
        this.successCb = []
        this.errorCb = []
        this._init(fn)
    }
    _init(fn) {
        const resolve = this._resolve.bind(this)
        const reject = this._reject.bind(this)
        fn(resolve, reject)
    }
    _resolve(value) {
        const self = this
        self.value = value
        self.state = 'fulfilled'
        setTimeout(() => {
            self.successCb.forEach((callback) => {
                callback(value)
            })
        }, 0)
    }
    _reject(reason) {
        const self = this
        self.value = reason
        self.state = 'rejected'
        setTimeout(() => {
            self.errorCb.forEach((callback) => {
                callback(reason)
            })
        }, 0)
    }
    then(fn) {
        if (this.state === 'pending') {
            this.successCb.push(fn)
        } else {
            const val = this.value
            fn(val)
        }
        return this
    }
    catch(fn) {
        if (this.state === 'pending') {
            this.errorCb.push(fn)
        } else {
            const val = this.value
            fn(val)
        }
        return this
    }
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市剂习,隨后出現(xiàn)的幾起案子蛮位,更是在濱河造成了極大的恐慌,老刑警劉巖鳞绕,帶你破解...
    沈念sama閱讀 211,042評論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件失仁,死亡現(xiàn)場離奇詭異,居然都是意外死亡猾昆,警方通過查閱死者的電腦和手機陶因,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 89,996評論 2 384
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來垂蜗,“玉大人楷扬,你說我怎么就攤上這事√” “怎么了烘苹?”我有些...
    開封第一講書人閱讀 156,674評論 0 345
  • 文/不壞的土叔 我叫張陵,是天一觀的道長片部。 經(jīng)常有香客問我镣衡,道長,這世上最難降的妖魔是什么档悠? 我笑而不...
    開封第一講書人閱讀 56,340評論 1 283
  • 正文 為了忘掉前任廊鸥,我火速辦了婚禮,結(jié)果婚禮上辖所,老公的妹妹穿的比我還像新娘惰说。我一直安慰自己,他們只是感情好缘回,可當我...
    茶點故事閱讀 65,404評論 5 384
  • 文/花漫 我一把揭開白布吆视。 她就那樣靜靜地躺著,像睡著了一般酥宴。 火紅的嫁衣襯著肌膚如雪啦吧。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,749評論 1 289
  • 那天拙寡,我揣著相機與錄音授滓,去河邊找鬼。 笑死,一個胖子當著我的面吹牛褒墨,可吹牛的內(nèi)容都是我干的炫刷。 我是一名探鬼主播,決...
    沈念sama閱讀 38,902評論 3 405
  • 文/蒼蘭香墨 我猛地睜開眼郁妈,長吁一口氣:“原來是場噩夢啊……” “哼浑玛!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起噩咪,我...
    開封第一講書人閱讀 37,662評論 0 266
  • 序言:老撾萬榮一對情侶失蹤顾彰,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后胃碾,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體涨享,經(jīng)...
    沈念sama閱讀 44,110評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,451評論 2 325
  • 正文 我和宋清朗相戀三年仆百,在試婚紗的時候發(fā)現(xiàn)自己被綠了厕隧。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 38,577評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡俄周,死狀恐怖吁讨,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情峦朗,我是刑警寧澤建丧,帶...
    沈念sama閱讀 34,258評論 4 328
  • 正文 年R本政府宣布勋篓,位于F島的核電站到忽,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏忌堂。R本人自食惡果不足惜尺铣,卻給世界環(huán)境...
    茶點故事閱讀 39,848評論 3 312
  • 文/蒙蒙 一拴曲、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧凛忿,春花似錦疗韵、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,726評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽流译。三九已至逞怨,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間福澡,已是汗流浹背叠赦。 一陣腳步聲響...
    開封第一講書人閱讀 31,952評論 1 264
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人除秀。 一個月前我還...
    沈念sama閱讀 46,271評論 2 360
  • 正文 我出身青樓糯累,卻偏偏與公主長得像,于是被迫代替她去往敵國和親册踩。 傳聞我的和親對象是個殘疾皇子泳姐,可洞房花燭夜當晚...
    茶點故事閱讀 43,452評論 2 348

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

  • Promise原理解析 標簽(空格分隔): Node.js Promise原理解析 簡介 Promise 對象用于...
    袁俊亮技術(shù)博客閱讀 3,366評論 2 5
  • 首先我們來看一個例子: 考慮下面一種獲取用戶id的請求處理 getUserId方法返回一個promise,可以通過...
    xiongshunshun閱讀 463評論 0 0
  • 淺灘濕腳剛洗腳暂吉, 遠離天敵曬美照胖秒。 孤身旅游清靜地, 趕緊去掉一身騷慕的。 一一彭甫2018年 9月11日清晨書
    彭甫閱讀 269評論 4 4
  • 最近結(jié)石病的患者又多了起來肮街,每天都有預約床位的患者风题,特別是其中還有一些熟面孔。人都知道日常要多喝水嫉父,不過不少人就有...
    大海_f916閱讀 368評論 0 0
  • 今晚兒子還沒回來沛硅,就有小伙伴來找他玩了,看來他們都玩的很愉快熔号。我在網(wǎng)上給兒子買了一個籃球稽鞭,兒子回來就拿著去和伙伴玩...
    六年級三班陳楓媽媽閱讀 246評論 0 1