淺析mobx-react源碼(一)—自動追蹤依賴

使用一種工具就像使用一個包裝好的黑盒子吃挑,我們不必探究其內(nèi)部到底是如何實現(xiàn)钝荡,只需要能夠?qū)⒂梅巳挥谛兀裁礃拥妮斎霑玫绞裁礃拥妮敵瞿苡型昝赖念A(yù)測儒鹿。但不幸的事化撕,大部分文檔都難以讓自己達(dá)到這點,為此我們不得不淺析一下源碼约炎,來探尋他是如何實現(xiàn)的植阴,避免出現(xiàn)意料之外的情況導(dǎo)致bug,也防止做出多余的操作圾浅,精簡代碼掠手,減少bug。

自動追蹤依賴

在使用redux的時候狸捕,我們不得不使用connect來從store中取到當(dāng)前組件所需要的state喷鸽,這其實也是一個依賴分析的問題,只有當(dāng)組件所依賴的state變化時灸拍,當(dāng)前組件才會更新做祝,避免了不必要的render砾省。而在使用mobx時,這一步被自動完成了混槐,因此在使用mobx時编兄,會感到極其酸爽舒適,只需要一個@observer就能盡情地使用store中的state了声登,而且完全不用擔(dān)心性能問題狠鸳,那么這究竟是如何做到的?

了解mobx機制

在探究mobx-react的源碼之前悯嗓,得先了解一下mobx件舵。

import { Reaction, observable } from 'mobx';
const obj = observable({ num: 0 });
const reaction = new Reaction('reaction name', function () {
  this.track(() => {
    console.log('track num++', obj.num);
  });
});
reaction.track(() => {
  console.log('reaction init', obj.num);
});
// 輸出 reaction init 0

obj.num++; // 輸出 track num++ 1
obj.num++; // 輸出 track num++ 2

mobx在observable改造了{ num: 0 }這個對象為可觀察對象,當(dāng)取obj的屬性num時脯厨,能會執(zhí)行其內(nèi)部的方法铅祸。
reaction的track方法傳進(jìn)去的回調(diào)中,執(zhí)行console.log('reaction init', obj.num)時俄认,get到了obj.num值个少,這時mobx會將這個屬性作為依賴與此reaction綁定,當(dāng)obj.num變化時眯杏,就會執(zhí)行實例化reaction時傳進(jìn)去進(jìn)去的回調(diào)

function () {
  this.track(() => {
    console.log('track num++', obj.num);
  });
}

這時會輸出track num++ 1夜焦,并從新確定依賴關(guān)系。接下來看下一個例子

import { Reaction, observable } from 'mobx';
const obj = observable({ num: 0, letter: 'a', bool: true });
const callback = function () {
  if (obj.bool) {
    console.log(obj.num);
  } else {
    console.log(obj.letter);
  }
};
const reaction = new Reaction('reaction name', function () {
  this.track(callback);
});
reaction.track(callback); // 第一次track岂贩,分析到依賴[obj.bool, obj.num]茫经,輸出 0
obj.num++; // 第二次track,分析到依賴依然是[obj.bool, obj.num]萎津,輸出 1
obj.letter = 'b'; // obj.letter不在依賴中卸伞,不執(zhí)行this.track(callback)
obj.bool = false; // obj.bool在依賴中岩梳,執(zhí)行this.track(callback);重新分析依賴[obj.bool, obj.letter] 輸出 b
obj.num++; // 此時obj.num不在依賴中欲间,不輸出任何值
obj.letter = 'c'; // 此時obj.letter在依賴中,輸出 c

由此可見烘绽,mobx可以在每次執(zhí)行時會分析依賴颈渊,并且將在callback中遂黍,卻并不執(zhí)行的屬性排除在依賴外,減少了callback俊嗽。
看到這里雾家,我們應(yīng)該能想到,只要將callback當(dāng)成react的render绍豁,就能完美地解決對react的對接芯咧,并且比使用redux的時候性能更好!因為使用redux的時候?qū)⒁蕾噉um,letter敬飒,bool三個屬性邪铲,無論哪個改變都講觸發(fā)組件的render,除非根據(jù)業(yè)務(wù)邏輯寫上復(fù)雜的componentShouldUpdate驶拱,而使用mobx時霜浴,只要其中某個state的改變不影響view的改變時晶衷,這個state的變化就不會引起組件render蓝纲,達(dá)到更細(xì)粒度上對render的控制,近乎完美地排除不需要的render晌纫。

mobx-react源碼片段探究

那么接下來看一看mobx-react的源碼是如何實現(xiàn)的

// 使用mobx-react
import React, { Component} from 'react';
import { observer } from 'mobx-react';
@observer
export default MyCom extends Component {
  render() {
    // ...do something
  }
}
// mobx-react的源碼片段
// ...
const baseRender = this.render.bind(this) // 組件自己的render就是baseRender
let reaction = null
let isRenderingPending = false
// 定義initialRender税迷,用來代替組件第一次render
const initialRender = () => {
  reaction = new Reaction(`${initialName}#${rootNodeID}.render()`, () => {
      if (!isRenderingPending) {
          // 防止在constructor中修改store觸發(fā)引起的副作用(此時還沒有初始render)
          isRenderingPending = true

          // 觸發(fā)mobx-react添加的一個生命周期
          if (typeof this.componentWillReact === "function") this.componentWillReact() 

          // 防止componentWillRect引起的副作用
          if (this.__$mobxIsUnmounted !== true) {
              let hasError = true
              try {
                  isForcingUpdate = true
                  // skipRender是用來防止死循環(huán),這里不用管
                  // 當(dāng)依賴的state有變化是會使用forceUpdate強制render
                  // 并解析新的依賴
                  if (!skipRender) Component.prototype.forceUpdate.call(this)
                  hasError = false
              } finally {
                  isForcingUpdate = false
                  if (hasError) reaction.dispose()
              }
          }
      }
  })
  reaction.reactComponent = this
  reactiveRender.$mobx = reaction
  // 用reactRender替代render
  this.render = reactiveRender
  // 執(zhí)行reactiveRender
  return reactiveRender()
}

const reactiveRender = () => {
  isRenderingPending = false
  let exception = undefined
  let rendering = undefined
  reaction.track(() => {
      if (isDevtoolsEnabled) {
          this.__$mobRenderStart = Date.now()
      }
      try {
          // 允許state改變锹漱,然后執(zhí)行baseRedner箭养,這里將解析一次render中對state屬性的依賴
          rendering = extras.allowStateChanges(false, baseRender)
      } catch (e) {
          exception = e
      }
      if (isDevtoolsEnabled) {
          this.__$mobRenderEnd = Date.now()
      }
  })
  if (exception) {
      errorsReporter.emit(exception)
      throw exception
  }
  return rendering
}
// ...

這樣就達(dá)到了用mobx控制react組件的render。

結(jié)論

對這一段代碼我們可以知道哥牍,僅僅在組件上加一行@observer然后使用mobx的可觀察屬性進(jìn)行render控制view毕泌,就能達(dá)到性能的高效化,對單個state屬性的粒度上控制組件的render嗅辣,所以放心大膽地用mobx來管理應(yīng)用的state吧撼泛!

下一篇 分批處理變化

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市澡谭,隨后出現(xiàn)的幾起案子愿题,更是在濱河造成了極大的恐慌,老刑警劉巖蛙奖,帶你破解...
    沈念sama閱讀 218,122評論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件潘酗,死亡現(xiàn)場離奇詭異,居然都是意外死亡雁仲,警方通過查閱死者的電腦和手機仔夺,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,070評論 3 395
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來攒砖,“玉大人缸兔,你說我怎么就攤上這事〖礼茫” “怎么了灶体?”我有些...
    開封第一講書人閱讀 164,491評論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀的道長掐暮。 經(jīng)常有香客問我蝎抽,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,636評論 1 293
  • 正文 為了忘掉前任樟结,我火速辦了婚禮养交,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘瓢宦。我一直安慰自己碎连,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 67,676評論 6 392
  • 文/花漫 我一把揭開白布驮履。 她就那樣靜靜地躺著鱼辙,像睡著了一般。 火紅的嫁衣襯著肌膚如雪玫镐。 梳的紋絲不亂的頭發(fā)上倒戏,一...
    開封第一講書人閱讀 51,541評論 1 305
  • 那天,我揣著相機與錄音恐似,去河邊找鬼杜跷。 笑死,一個胖子當(dāng)著我的面吹牛矫夷,可吹牛的內(nèi)容都是我干的葛闷。 我是一名探鬼主播,決...
    沈念sama閱讀 40,292評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼双藕,長吁一口氣:“原來是場噩夢啊……” “哼淑趾!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起蔓彩,我...
    開封第一講書人閱讀 39,211評論 0 276
  • 序言:老撾萬榮一對情侶失蹤治笨,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后赤嚼,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體旷赖,經(jīng)...
    沈念sama閱讀 45,655評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,846評論 3 336
  • 正文 我和宋清朗相戀三年更卒,在試婚紗的時候發(fā)現(xiàn)自己被綠了等孵。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,965評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡蹂空,死狀恐怖俯萌,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情上枕,我是刑警寧澤咐熙,帶...
    沈念sama閱讀 35,684評論 5 347
  • 正文 年R本政府宣布,位于F島的核電站辨萍,受9級特大地震影響棋恼,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,295評論 3 329
  • 文/蒙蒙 一爪飘、第九天 我趴在偏房一處隱蔽的房頂上張望义起。 院中可真熱鬧,春花似錦师崎、人聲如沸默终。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,894評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽齐蔽。三九已至,卻和暖如春昼汗,著一層夾襖步出監(jiān)牢的瞬間肴熏,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,012評論 1 269
  • 我被黑心中介騙來泰國打工顷窒, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人源哩。 一個月前我還...
    沈念sama閱讀 48,126評論 3 370
  • 正文 我出身青樓鞋吉,卻偏偏與公主長得像,于是被迫代替她去往敵國和親励烦。 傳聞我的和親對象是個殘疾皇子谓着,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,914評論 2 355

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