mobx-react 源碼閱讀筆記(一)

前言

emm 由于實習(xí)的項目組只有自己一個人,選型開發(fā)都為所欲為了tat,從以redux進(jìn)行數(shù)據(jù)流管理的使用跨释,決定嘗試一波實戰(zhàn)mobx送丰,在使用中不禁在想酿傍,mobx是怎么和react連在一起炉媒,而mobx又是為什么可以如此高效便捷踪区。react-redux的驅(qū)動視圖更新是通過connect組件,在外層HOC中訂閱store改變觸發(fā)
setState({})進(jìn)行更新視圖。而mobx-react中的observer呢橱野?

這里可以看到最關(guān)鍵的一點,區(qū)別于redux的更新方式,在mobx-react中是直接使用forceUpdate方法進(jìn)行更新視圖,而區(qū)別于redux 對store的訂閱,mobx-react又是怎么樣觸發(fā)forceUpdate呢?

/**
 *強(qiáng)制更新組件https://github.com/mobxjs/mobx-react/blob/master/src/observer.js#L188
 */
Component.prototype.forceUpdate.call(this)

定位到observer,先了解observer執(zhí)行的流程


1.png

核心點


2.png

talk is so cheap show you the code

//... 
// https://github.com/mobxjs/mobx-react/blob/master/src/observer.js#L334
mixinLifecycleEvents(target)
//mixin組件的生命周期,將reactiveMixin中的函數(shù)mixin到具體的生命周期中.

在mixin中最核心需要關(guān)注的就是componentWillMount,
先簡單提一下mixinLifecycleEvents區(qū)別對待的是shouldComponentUpdate,如果該方法沒有定義observer會直接將其重寫為PureComponentshouldComponentUpdate的實現(xiàn)

進(jìn)一步看componentWillMount

//https://github.com/mobxjs/mobx-react/blob/master/src/observer.js#L168
// wire up reactive render
        //保存當(dāng)前函數(shù)的render
        const baseRender = this.render.bind(this)
        let reaction = null
        let isRenderingPending = false

        const initialRender = () => {
            /*綁定reaction善玫,observable屬性改變的時候會觸發(fā)這個(mobx實質(zhì)是雙向綁定水援,observable更新視圖也要更新,這里是數(shù)據(jù)綁定到視圖上的第一步茅郎。)
            */
            reaction = new Reaction(`${initialName}#${rootNodeID}.render()`, () => {
                if (!isRenderingPending) {
                    // N.B. Getting here *before mounting* means that a component constructor has side effects (see the relevant test in misc.js)
                    // This unidiomatic React usage but React will correctly warn about this so we continue as usual
                    // See #85 / Pull #44
                    isRenderingPending = true
                    if (typeof this.componentWillReact === "function") this.componentWillReact() // TODO: wrap in action?
                    if (this.__$mobxIsUnmounted !== true) {
                        // If we are unmounted at this point, componentWillReact() had a side effect causing the component to unmounted
                        // TODO: remove this check? Then react will properly warn about the fact that this should not happen? See #73
                        // However, people also claim this migth happen during unit tests..
                        let hasError = true
                        try {
                            isForcingUpdate = true
                            if (!skipRender) Component.prototype.forceUpdate.call(this)
                            hasError = false
                        } finally {
                            isForcingUpdate = false
                            if (hasError) reaction.dispose()
                        }
                    }
                }
            })
            reaction.reactComponent = this
            reactiveRender.$mobx = reaction
            // 重寫render
            this.render = reactiveRender
            // 實際的render
            return reactiveRender()
        }
        
        const reactiveRender = () => {
            isRenderingPending = false
            let exception = undefined
            let rendering = undefined
            /**
            * 核心關(guān)聯(lián)部分
            * 追蹤數(shù)據(jù)  
            * https://github.com/mobxjs/mobx/blob/master/src/core/reaction.ts#L112
            * reaction.track(fn: () => void)
            * 實際上
            * trackDerivedFunction<T>(derivation: IDerivation, f: () => T, context)
            * const result = trackDerivedFunction(this, fn, undefined)// this對應(yīng)reaction,fn對應(yīng)track中的fn
            *   (https://github.com/mobxjs/mobx/blob/master/src/core/derivation.ts#L131)
            * trackDerivedFunction這個函數(shù)有什么作用蜗元? 
            * 執(zhí)行函數(shù)f并跟蹤那些可觀察并且正在f函數(shù)中引用的變量,將這些可追蹤的變量注冊并儲存在derivation中即reaction中
            * 
            * f中引用的變量 核心上是通過atom.reportObserved()關(guān)聯(lián)引用
            * 簡單例子見   makePropertyObservableReference  中的      
            *       get: function() {
            *        atom.reportObserved() 
            *        return valueHolder
            *    },
            * 
            *  重新到trackDerivedFunction執(zhí)行中
            * result = f.call(context); 
            *f本身已經(jīng)是箭頭函數(shù)了,上下文已經(jīng)綁定過了. 這里實際上就是執(zhí)行component的render,
            *實際上就是通過這里收集到observable的引用 (依賴收集)
            *trackDerivedFunction中會derivation進(jìn)行屬性更新,以及通過bindDependencies更新依賴收集的情況系冗。
            **/
            reaction.track(() => {
                if (isDevtoolsEnabled) {
                    this.__$mobRenderStart = Date.now()
                }
                try {
                    rendering = extras.allowStateChanges(false, baseRender)
                } catch (e) {
                    exception = e
                }
                if (isDevtoolsEnabled) {
                    this.__$mobRenderEnd = Date.now()
                }
            })
            if (exception) {
                errorsReporter.emit(exception)
                throw exception
            }
            return rendering
        }

        this.render = initialRender
    },

這一次的閱讀之行到此就暫告一段落了,關(guān)于derivation原理的進(jìn)一步理解需要進(jìn)一步學(xué)習(xí)mobx的源碼奕扣。目前只是有較淺的理解,如有不正之處,請大家指出掌敬! tat 畢竟只是閉門造車惯豆。


注釋版源碼

學(xué)習(xí)資料

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市奔害,隨后出現(xiàn)的幾起案子楷兽,更是在濱河造成了極大的恐慌,老刑警劉巖华临,帶你破解...
    沈念sama閱讀 206,311評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件芯杀,死亡現(xiàn)場離奇詭異,居然都是意外死亡雅潭,警方通過查閱死者的電腦和手機(jī)揭厚,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,339評論 2 382
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來扶供,“玉大人筛圆,你說我怎么就攤上這事〈慌ǎ” “怎么了顽染?”我有些...
    開封第一講書人閱讀 152,671評論 0 342
  • 文/不壞的土叔 我叫張陵,是天一觀的道長轰绵。 經(jīng)常有香客問我粉寞,道長,這世上最難降的妖魔是什么左腔? 我笑而不...
    開封第一講書人閱讀 55,252評論 1 279
  • 正文 為了忘掉前任唧垦,我火速辦了婚禮,結(jié)果婚禮上液样,老公的妹妹穿的比我還像新娘振亮。我一直安慰自己巧还,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 64,253評論 5 371
  • 文/花漫 我一把揭開白布坊秸。 她就那樣靜靜地躺著麸祷,像睡著了一般。 火紅的嫁衣襯著肌膚如雪褒搔。 梳的紋絲不亂的頭發(fā)上阶牍,一...
    開封第一講書人閱讀 49,031評論 1 285
  • 那天,我揣著相機(jī)與錄音星瘾,去河邊找鬼走孽。 笑死,一個胖子當(dāng)著我的面吹牛琳状,可吹牛的內(nèi)容都是我干的磕瓷。 我是一名探鬼主播,決...
    沈念sama閱讀 38,340評論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼念逞,長吁一口氣:“原來是場噩夢啊……” “哼困食!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起翎承,我...
    開封第一講書人閱讀 36,973評論 0 259
  • 序言:老撾萬榮一對情侶失蹤陷舅,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后审洞,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體莱睁,經(jīng)...
    沈念sama閱讀 43,466評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 35,937評論 2 323
  • 正文 我和宋清朗相戀三年芒澜,在試婚紗的時候發(fā)現(xiàn)自己被綠了仰剿。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 38,039評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡痴晦,死狀恐怖南吮,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情誊酌,我是刑警寧澤部凑,帶...
    沈念sama閱讀 33,701評論 4 323
  • 正文 年R本政府宣布,位于F島的核電站碧浊,受9級特大地震影響涂邀,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜箱锐,卻給世界環(huán)境...
    茶點故事閱讀 39,254評論 3 307
  • 文/蒙蒙 一比勉、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦浩聋、人聲如沸观蜗。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,259評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽墓捻。三九已至,卻和暖如春坊夫,著一層夾襖步出監(jiān)牢的瞬間砖第,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,485評論 1 262
  • 我被黑心中介騙來泰國打工践樱, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留厂画,地道東北人凸丸。 一個月前我還...
    沈念sama閱讀 45,497評論 2 354
  • 正文 我出身青樓拷邢,卻偏偏與公主長得像,于是被迫代替她去往敵國和親屎慢。 傳聞我的和親對象是個殘疾皇子瞭稼,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 42,786評論 2 345

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