Vue3源碼--響應式原理3(應用)

?前兩篇寫了響應式系統(tǒng)的兩個核心模塊effect,reactivity宏侍,這篇寫一下響應式系統(tǒng)在源碼中的應用吧潮太。

Computed API

?話不多說,首先來看一下利用響應式系統(tǒng)實現computed api谆构,直接上代碼注釋裸扶。

// 3.x的computed 支持設置setter,setter的邏輯很簡單搬素,就不多說了呵晨,這里刪掉了相關的代碼。
// 大部分的情況下傳入的是一個function, 也就是getter
export function computed<T>(
  getterOrOptions: ComputedGetter<T> | WritableComputedOptions<T>
) {
  let getter: ComputedGetter<T>

  let dirty = true
  let value: T
  let computed: ComputedRef<T>
  // 每個computed中都會維護一個lazy effect實例
  const runner = effect(getter, {
    lazy: true,
    // mark effect as computed so that it gets priority during trigger
    computed: true,
    // 注意:這個scheduler并不執(zhí)行effect熬尺,僅僅把dirty設置為true
    scheduler: () => {
      if (!dirty) {
        dirty = true
        // 觸發(fā)依賴這個computed的effects
        trigger(computed, TriggerOpTypes.SET, 'value')
      }
    }
  })
  computed = {
    _isRef: true,
    // expose effect so computed can be stopped
    effect: runner,
    get value() {
      // 如果數據是臟的摸屠,就執(zhí)行自身的effect獲取最新數據,否則直接返回緩存的數據
      if (dirty) {
        value = runner()
        dirty = false
      }
      // 此時如果有active effect的話粱哼,就會被收集到computed的depsMap里面
      track(computed, TrackOpTypes.GET, 'value')
      return value
    }
  } as any
  return computed
}

我試著用語言描述一下:

  1. computed和ref一樣季二,返回一個封裝的對象,實際上他的_isRef就是true
  2. 內部維護一個lazy effect,之前也講過胯舷,lazy effect不會立刻執(zhí)行
  3. 在第一次獲取這個computed的值的時候刻蚯,執(zhí)行這個lazy effect,此時active effect就是這個lazy effect桑嘶,computed里面的響應式數據都會收集到這個lazy effect炊汹,
  4. 當computed的響應式數據發(fā)生改變,由于設置了scheduler不翩,并不會去執(zhí)行這個lazy effect兵扬,而是把這個computed標示為臟,然后直接trigger這個computed下所有的effect口蝠。(如果computed是在render中調用的話器钟,那么就相當于重新去執(zhí)行render的過程)

Watch API

? 去看watch api的源碼packages/runtime-core/src/apiWatch.ts, 會發(fā)現好長,其實watch api的原理最簡單了妙蔗,effect方法的作用是監(jiān)聽傳入的函數傲霸,如果響應式數據發(fā)生改變,那么就重新執(zhí)行這個函數眉反,而effect方法支持scheduler配置昙啄,如果傳入了scheduler方法,就不會直接重新執(zhí)行這個effect,我們只需要在shceduler方法里面去執(zhí)行watch api的回調即可寸五。

?終于寫完響應式原理相關的了梳凛,發(fā)現寫這種源碼解讀的文章是真難,不過這個過程需要組織語言去描述梳杏,自己還是加深了理解韧拒。后面接下去本來打算寫compile相關的內容,但是感覺如果像響應式這樣寫的話也太難面面俱到了十性,也沒有那么多精力叛溢。因此打算最后寫幾篇,內容就是vue3.x相比2.x快在哪里劲适,做了哪些優(yōu)化楷掉。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市霞势,隨后出現的幾起案子烹植,更是在濱河造成了極大的恐慌,老刑警劉巖愕贡,帶你破解...
    沈念sama閱讀 218,122評論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件刊橘,死亡現場離奇詭異,居然都是意外死亡颂鸿,警方通過查閱死者的電腦和手機促绵,發(fā)現死者居然都...
    沈念sama閱讀 93,070評論 3 395
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人败晴,你說我怎么就攤上這事浓冒。” “怎么了尖坤?”我有些...
    開封第一講書人閱讀 164,491評論 0 354
  • 文/不壞的土叔 我叫張陵稳懒,是天一觀的道長。 經常有香客問我慢味,道長场梆,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,636評論 1 293
  • 正文 為了忘掉前任纯路,我火速辦了婚禮或油,結果婚禮上,老公的妹妹穿的比我還像新娘驰唬。我一直安慰自己顶岸,他們只是感情好,可當我...
    茶點故事閱讀 67,676評論 6 392
  • 文/花漫 我一把揭開白布叫编。 她就那樣靜靜地躺著辖佣,像睡著了一般。 火紅的嫁衣襯著肌膚如雪搓逾。 梳的紋絲不亂的頭發(fā)上卷谈,一...
    開封第一講書人閱讀 51,541評論 1 305
  • 那天,我揣著相機與錄音霞篡,去河邊找鬼雏搂。 笑死,一個胖子當著我的面吹牛寇损,可吹牛的內容都是我干的。 我是一名探鬼主播裳食,決...
    沈念sama閱讀 40,292評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼矛市,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了诲祸?” 一聲冷哼從身側響起浊吏,我...
    開封第一講書人閱讀 39,211評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎救氯,沒想到半個月后找田,有當地人在樹林里發(fā)現了一具尸體,經...
    沈念sama閱讀 45,655評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡着憨,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 37,846評論 3 336
  • 正文 我和宋清朗相戀三年墩衙,在試婚紗的時候發(fā)現自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,965評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡漆改,死狀恐怖心铃,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情挫剑,我是刑警寧澤去扣,帶...
    沈念sama閱讀 35,684評論 5 347
  • 正文 年R本政府宣布,位于F島的核電站樊破,受9級特大地震影響愉棱,放射性物質發(fā)生泄漏。R本人自食惡果不足惜哲戚,卻給世界環(huán)境...
    茶點故事閱讀 41,295評論 3 329
  • 文/蒙蒙 一奔滑、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧惫恼,春花似錦档押、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,894評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至腕窥,卻和暖如春粒没,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背簇爆。 一陣腳步聲響...
    開封第一講書人閱讀 33,012評論 1 269
  • 我被黑心中介騙來泰國打工癞松, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人入蛆。 一個月前我還...
    沈念sama閱讀 48,126評論 3 370
  • 正文 我出身青樓响蓉,卻偏偏與公主長得像,于是被迫代替她去往敵國和親哨毁。 傳聞我的和親對象是個殘疾皇子枫甲,可洞房花燭夜當晚...
    茶點故事閱讀 44,914評論 2 355

推薦閱讀更多精彩內容