国产xxxx99真实实拍_久久不雅视频_高清韩国a级特黄毛片_嗯老师别我我受不了了小说

資訊專(zhuān)欄INFORMATION COLUMN

【React進(jìn)階系列】 虛擬dom與diff算法

zhou_you / 2705人閱讀

摘要:通過(guò)對(duì)樹(shù)進(jìn)行層級(jí)控制,同一個(gè)父節(jié)點(diǎn)下的所有子節(jié)點(diǎn)。新老集合進(jìn)行差異化對(duì)比,發(fā)現(xiàn),則創(chuàng)建并插入至新集合,刪除老集合以此類(lèi)推,創(chuàng)建并插入和,刪除和。

虛擬dom

Jsx 表面寫(xiě)的是html,其實(shí)內(nèi)部執(zhí)行的是一段js
createElement

React.createElement(
  type,
  [props],
  [...children]
)

createElement把這個(gè)樹(shù)形結(jié)構(gòu),存在內(nèi)存里面
Jsx最終以這樣的一個(gè)個(gè)對(duì)象遞歸的存在內(nèi)存中,執(zhí)行diff算法

多層結(jié)構(gòu)

簡(jiǎn)單的createElement實(shí)現(xiàn)


reactElement - 生成的是一個(gè)對(duì)象來(lái)描述這個(gè)節(jié)點(diǎn)

react diff

與傳統(tǒng)樹(shù)的diff的區(qū)別

計(jì)算一棵樹(shù)形結(jié)構(gòu)轉(zhuǎn)換成另一棵樹(shù)形結(jié)構(gòu)的最少操作,是一個(gè)復(fù)雜且值得研究的問(wèn)題。傳統(tǒng) diff 算法通過(guò)循環(huán)遞歸對(duì)節(jié)點(diǎn)進(jìn)行依次對(duì)比,效率低下,算法復(fù)雜度達(dá)到 O(n^3)

react diff策略

Web UI 中 DOM 節(jié)點(diǎn)跨層級(jí)的移動(dòng)操作特別少,可以忽略不計(jì)。

擁有相同類(lèi)的兩個(gè)組件將會(huì)生成相似的樹(shù)形結(jié)構(gòu),擁有不同類(lèi)的兩個(gè)組件將會(huì)生成不同的樹(shù)形結(jié)構(gòu)。

對(duì)于同一層級(jí)的一組子節(jié)點(diǎn),它們可以通過(guò)唯一 id 進(jìn)行區(qū)分。

tree diff

 基于策略一,對(duì)樹(shù)進(jìn)行分層比較,兩棵樹(shù)只會(huì)對(duì)同一層次的節(jié)點(diǎn)進(jìn)行比較。  
 React 通過(guò) updateDepth 對(duì) Virtual DOM 樹(shù)進(jìn)行層級(jí)控制,同一個(gè)父節(jié)點(diǎn)下的所有子節(jié)點(diǎn)。
 

什么是 DOM 節(jié)點(diǎn)跨層級(jí)的移動(dòng)操作?

A 節(jié)點(diǎn)(包括其子節(jié)點(diǎn))整個(gè)被移動(dòng)到 D 節(jié)點(diǎn)下

如果出現(xiàn)了 DOM 節(jié)點(diǎn)跨層級(jí)的移動(dòng)操作,React diff 會(huì)有怎樣的表現(xiàn)呢?

React 只會(huì)簡(jiǎn)單的考慮同層級(jí)節(jié)點(diǎn)的位置變換,而對(duì)于不同層級(jí)的節(jié)點(diǎn),只有創(chuàng)建和刪除操作。

當(dāng)根節(jié)點(diǎn)發(fā)現(xiàn)子節(jié)點(diǎn)中 A 消失了,就會(huì)直接銷(xiāo)毀 A;當(dāng) D 發(fā)現(xiàn)多了一個(gè)子節(jié)點(diǎn) A,則會(huì)創(chuàng)建新的 A(包括子節(jié)點(diǎn))作為其子節(jié)點(diǎn)。此時(shí),React diff 的執(zhí)行情況:create A -> create B -> create C -> delete A。

注意:

在開(kāi)發(fā)組件時(shí),保持穩(wěn)定的 DOM 結(jié)構(gòu)會(huì)有助于性能的提升。例如,可以通過(guò) CSS 隱藏或顯示節(jié)點(diǎn),而不是真的移除或添加 DOM 節(jié)點(diǎn)。

component diff

依據(jù)策略二

如果是同一類(lèi)型的組件,按照原策略繼續(xù)比較 virtual DOM tree。

如果不是,則將該組件判斷為 dirty component,從而替換整個(gè)組件下的所有子節(jié)點(diǎn)。

對(duì)于同一類(lèi)型的組件,有可能其 Virtual DOM 沒(méi)有任何變化,如果能夠確切的知道這點(diǎn)那可以節(jié)省大量的 diff 運(yùn)算時(shí)間,因此 React 允許用戶(hù)通過(guò) shouldComponentUpdate() 來(lái)判斷該組件是否需要進(jìn)行 diff。

React 判斷 D 和 G 是不同類(lèi)型的組件,就不會(huì)比較二者的結(jié)構(gòu),而是直接刪除 component D,重新創(chuàng)建 component G 以及其子節(jié)點(diǎn),即使D 和 G的結(jié)構(gòu)很相似

element diff

當(dāng)節(jié)點(diǎn)處于同一層級(jí)時(shí),React diff 提供了三種節(jié)點(diǎn)操作,分別為:INSERT_MARKUP(插入)、MOVE_EXISTING(移動(dòng))和 REMOVE_NODE(刪除)。

INSERT_MARKUP,新的 component 類(lèi)型不在老集合里, 即是全新的節(jié)點(diǎn),需要對(duì)新節(jié)點(diǎn)執(zhí)行插入操作。

MOVE_EXISTING,在老集合有新 component 類(lèi)型,且 element 是可更新的類(lèi)型,generateComponentChildren 已調(diào)用 receiveComponent,這種情況下 prevChild=nextChild,就需要做移動(dòng)操作,可以復(fù)用以前的 DOM 節(jié)點(diǎn)。

REMOVE_NODE,老 component 類(lèi)型,在新集合里也有,但對(duì)應(yīng)的 element 不同則不能直接復(fù)用和更新,需要執(zhí)行刪除操作,或者老 component 不在新集合里的,也需要執(zhí)行刪除操作。

eg: 新老集合進(jìn)行 diff 差異化對(duì)比,發(fā)現(xiàn) B != A,則創(chuàng)建并插入 B 至新集合,刪除老集合 A;以此類(lèi)推,創(chuàng)建并插入 A、D 和 C,刪除 B、C 和 D。

帶來(lái)的問(wèn)題:都是相同的節(jié)點(diǎn),但由于位置發(fā)生變化,導(dǎo)致需要進(jìn)行繁雜低效的刪除、創(chuàng)建操作,其實(shí)只要對(duì)這些節(jié)點(diǎn)進(jìn)行位置移動(dòng)即可

react優(yōu)化策略:允許開(kāi)發(fā)者對(duì)同一層級(jí)的同組子節(jié)點(diǎn),添加唯一 key 進(jìn)行區(qū)分

優(yōu)化后diff實(shí)現(xiàn):

對(duì)新集合的節(jié)點(diǎn)進(jìn)行循環(huán)遍歷,通過(guò)唯一 key 可以判斷新老集合中是否存在相同的節(jié)點(diǎn)

如果存在相同節(jié)點(diǎn),則進(jìn)行移動(dòng)操作,但在移動(dòng)前需要將當(dāng)前節(jié)點(diǎn)在老集合中的位置child._mountIndex與lastIndex(訪問(wèn)過(guò)的節(jié)點(diǎn)在老集合中最右的位置即最大的位置)進(jìn)行比較,if (child._mountIndex < lastIndex),則進(jìn)行節(jié)點(diǎn)移動(dòng)操作

分析:

element  _mountIndex  lastIndex  nextIndex  enqueueMove
B        1            0          0          false
A        0            1          1          true
D        3            1          2          false
C        2            3          3          true

step:

從新集合中取得 B,判斷老集合中存在相同節(jié)點(diǎn) B
B 在老集合中的位置 B._mountIndex = 1
初始 lastIndex = 0
不滿足 child._mountIndex < lastIndex 的條件,因此不對(duì) B 進(jìn)行移動(dòng)操作
更新 lastIndex = Math.max(prevChild._mountIndex, lastIndex) lastIndex更新為1
將 B 的位置更新為新集合中的位置prevChild._mountIndex = nextIndex,此時(shí)新集合中 B._mountIndex = 0,nextIndex++

以上主要分析新老集合中存在相同節(jié)點(diǎn)但位置不同時(shí),對(duì)節(jié)點(diǎn)進(jìn)行位置移動(dòng)的情況,如果新集合中有新加入的節(jié)點(diǎn)且老集合存在需要?jiǎng)h除的節(jié)點(diǎn),那么 React diff 又是如何對(duì)比運(yùn)作的呢?

    element  _mountIndex  lastIndex  nextIndex  enqueueMove
    B        1            0          0          false
    E        no exist
    C        2            1          2          false
    A        0            2          3          true
    

step

新建:從新集合中取得 E,判斷老集合中不存在相同節(jié)點(diǎn) E,則創(chuàng)建新節(jié)點(diǎn) E
     lastIndex不做處理
     E 的位置更新為新集合中的位置,nextIndex++
刪除:當(dāng)完成新集合中所有節(jié)點(diǎn) diff 時(shí),最后還需要對(duì)老集合進(jìn)行循環(huán)遍歷,判斷是否存在新集合中沒(méi)有但老集合中仍存在的節(jié)點(diǎn),發(fā)現(xiàn)存在這樣的節(jié)點(diǎn) D,因此刪除節(jié)點(diǎn) D

react diff的問(wèn)題

理論上 diff 應(yīng)該只需對(duì) D 執(zhí)行移動(dòng)操作,然而由于 D 在老集合的位置是最大的,導(dǎo)致其他節(jié)點(diǎn)的 _mountIndex < lastIndex,造成 D 沒(méi)有執(zhí)行移動(dòng)操作,而是 A、B、C 全部移動(dòng)到 D 節(jié)點(diǎn)后面的現(xiàn)象

建議:在開(kāi)發(fā)過(guò)程中,盡量減少類(lèi)似將最后一個(gè)節(jié)點(diǎn)移動(dòng)到列表首部的操作,當(dāng)節(jié)點(diǎn)數(shù)量過(guò)大或更新操作過(guò)于頻繁時(shí),在一定程度上會(huì)影響 React 的渲染性能。

總結(jié):

React 通過(guò)制定大膽的 diff 策略,將 O(n3) 復(fù)雜度的問(wèn)題轉(zhuǎn)換成 O(n) 復(fù)雜度的問(wèn)題;

React 通過(guò)分層求異的策略,對(duì) tree diff 進(jìn)行算法優(yōu)化;

React 通過(guò)相同類(lèi)生成相似樹(shù)形結(jié)構(gòu),不同類(lèi)生成不同樹(shù)形結(jié)構(gòu)的策略,對(duì) component diff 進(jìn)行算法優(yōu)化;

React 通過(guò)設(shè)置唯一 key的策略,對(duì) element diff 進(jìn)行算法優(yōu)化;

建議,在開(kāi)發(fā)組件時(shí),保持穩(wěn)定的 DOM 結(jié)構(gòu)會(huì)有助于性能的提升;

建議,在開(kāi)發(fā)過(guò)程中,盡量減少類(lèi)似將最后一個(gè)節(jié)點(diǎn)移動(dòng)到列表首部的操作,當(dāng)節(jié)點(diǎn)數(shù)量過(guò)大或更新操作過(guò)于頻繁時(shí),在一定程度上會(huì)影響 React 的渲染性能。

https://zhuanlan.zhihu.com/p/...

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://m.specialneedsforspecialkids.com/yun/98461.html

相關(guān)文章

  • React系列 --- virtualdom diff算法實(shí)現(xiàn)分析(三)

    摘要:所以只針對(duì)同層級(jí)節(jié)點(diǎn)做比較,將復(fù)雜度的問(wèn)題轉(zhuǎn)換成復(fù)雜度的問(wèn)題。 React系列 React系列 --- 簡(jiǎn)單模擬語(yǔ)法(一)React系列 --- Jsx, 合成事件與Refs(二)React系列 --- virtualdom diff算法實(shí)現(xiàn)分析(三)React系列 --- 從Mixin到HOC再到HOOKS(四)React系列 --- createElement, ReactElem...

    sunsmell 評(píng)論0 收藏0
  • 從零開(kāi)始實(shí)現(xiàn)一個(gè)React(三):diff算法

    摘要:而對(duì)比變化,找出需要更新部分的算法我們稱(chēng)之為算法。整個(gè)系列大概會(huì)有四篇,我每周會(huì)更新一到兩篇,我會(huì)第一時(shí)間在上更新,有問(wèn)題需要探討也請(qǐng)?jiān)谏匣貜?fù)我博客地址關(guān)注點(diǎn),訂閱點(diǎn)上一篇文章從零開(kāi)始實(shí)現(xiàn)一個(gè)二組件和生命周期 前言 在上一篇文章,我們已經(jīng)實(shí)現(xiàn)了React的組件功能,從功能的角度來(lái)說(shuō)已經(jīng)實(shí)現(xiàn)了React的核心功能了。 但是我們的實(shí)現(xiàn)方式有很大的問(wèn)題:每次更新都重新渲染整個(gè)應(yīng)用或者整個(gè)組件...

    The question 評(píng)論0 收藏0
  • React進(jìn)階系列】從零開(kāi)始手把手教你實(shí)現(xiàn)一個(gè)Virtual DOM(一)

    摘要:可實(shí)際上并不是創(chuàng)造的,將這個(gè)概念拿過(guò)來(lái)以后融會(huì)貫通慢慢地成為目前前端最炙手可熱的框架之一。則是將再抽象一層生成的簡(jiǎn)化版對(duì)象,這個(gè)對(duì)象也擁有上的一些屬性,比如等,但它是完全脫離于瀏覽器而存在的。所以今天我要手把手教大家怎么從零開(kāi)始實(shí)現(xiàn)。 假如你的項(xiàng)目使用了React,你知道怎么做性能優(yōu)化嗎?你知道為什么React讓你寫(xiě)shouldComponentUpdate或者React.PureCo...

    PumpkinDylan 評(píng)論0 收藏0
  • react進(jìn)階漫談

    摘要:父組件向子組件之間非常常見(jiàn),通過(guò)機(jī)制傳遞即可。我們應(yīng)該聽(tīng)說(shuō)過(guò)高階函數(shù),這種函數(shù)接受函數(shù)作為輸入,或者是輸出一個(gè)函數(shù),比如以及等函數(shù)。在傳遞數(shù)據(jù)的時(shí)候,我們可以用進(jìn)一步提高性能。 本文主要談自己在react學(xué)習(xí)的過(guò)程中總結(jié)出來(lái)的一些經(jīng)驗(yàn)和資源,內(nèi)容邏輯參考了深入react技術(shù)棧一書(shū)以及網(wǎng)上的諸多資源,但也并非完全照抄,代碼基本都是自己實(shí)踐,主要為平時(shí)個(gè)人學(xué)習(xí)做一個(gè)總結(jié)和參考。 本文的關(guān)鍵...

    neuSnail 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<