成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

Virtual DOM

BakerJ / 1547人閱讀

摘要:但瀏覽器沒這么智能,收到第一個更新請求后,并不知道后續(xù)還有次更新操作,因此會馬上執(zhí)行流程,最終執(zhí)行次流程。從拿出當(dāng)前節(jié)點的差異深度遍歷子節(jié)點對當(dāng)前節(jié)點進(jìn)行操作,根據(jù)不同類型的差異對當(dāng)前節(jié)點進(jìn)行操作結(jié)語算法主要是實現(xiàn)上面步驟的三個函數(shù),,。

大三戰(zhàn)五渣的我,平時也就只能用用別人的輪子,可總用不順心,畢竟不知道原理,最近用vue寫項目,里面涉及到的Virtual DOM雖然已不是什么新概念,但我也只是聽說而已,不知其所以然,既然看到大佬們解析后,那就記錄下吧
參考資料:
戴嘉華:https://github.com/livoras/bl...
張歆琳:https://www.jianshu.com/p/616...
王沛:https://www.infoq.cn/article/...

為啥要Virtual DOM

首先先了解一下加載一個HTML會發(fā)生哪些事情

使用HTML分析器生成DOM Tree

使用CSS分析器生成CSSOM

運(yùn)行JS

結(jié)合DOM Tree和CSSOM生成一棵Render Tree

根據(jù)render樹,瀏覽器可以計算出網(wǎng)頁中有哪些節(jié)點,各節(jié)點的CSS以及從屬關(guān)系,然后可以計算出每個節(jié)點在屏幕中的位置;

繪制出頁面

當(dāng)你用傳統(tǒng)的源生api或jQuery去操作DOM時,瀏覽器會從構(gòu)建DOM樹開始從頭到尾執(zhí)行一遍流程。比如當(dāng)你在一次操作時,需要更新10個DOM節(jié)點,理想狀態(tài)是一次性構(gòu)建完DOM樹,再執(zhí)行后續(xù)操作。但瀏覽器沒這么智能,收到第一個更新DOM請求后,并不知道后續(xù)還有9次更新操作,因此會馬上執(zhí)行流程,最終執(zhí)行10次流程。顯然例如計算DOM節(jié)點的坐標(biāo)值等都是白白浪費(fèi)性能,可能這次計算完,緊接著的下一個DOM更新請求,這個節(jié)點的坐標(biāo)值就變了,前面的一次計算是無用功。
DOM是很慢的,我們可以打印一下一個簡單的div元素的屬性


這還只是一層而已,真實的DOM會更加龐大,輕微的觸碰可能就會導(dǎo)致頁面重排,這可是殺死性能的罪魁禍?zhǔn)?。而相對于操作DOM對象,原生的JS對象處理起來更快而且簡單

步驟

JS表示DOM→構(gòu)建DOM樹→插圖文檔中

狀態(tài)變化→重新構(gòu)造一顆新的對象樹→新舊樹比較→記錄兩棵樹的差異

把2所記錄的差異應(yīng)用到步驟1所構(gòu)建的真正的DOM樹上,從而視圖更新了

Virtual DOM 的本質(zhì)

在 JS 和 DOM 之間做了一個緩存。可以類比 CPU 和硬盤,既然硬盤這么慢,我們就在它們之間加個緩存:既然 DOM 這么慢,我們就在它們 JS 和 DOM 之間加個緩存。CPU(JS)只操作內(nèi)存(Virtual DOM),最后的時候再把變更寫入硬盤(DOM)。

算法實現(xiàn) 步驟一:用JS對象模擬DOM樹

用JS記錄節(jié)點的類型,屬性和子節(jié)點
element.js

function Element (tagName, props, children) {
  this.tagName = tagName
  this.props = props
  this.children = children
}

function el(tagName, props, children){
  return new Element(tagName, props, children)
}

例如上面的 DOM 結(jié)構(gòu)就可以簡單的表示:

let el = require("./element")

let div= el("div", {id: "blue-div"}, [
  el("p", {class: "pink-p"}, [
    el("span", {class: "yellow-sapn"}, ["Virtual sapn"])]),
  el("ul", {class: "green-ul"}, [
    el("li", {class: "red-li"}, ["Virtual li1"]),
    el("li", {class: "red-li"}, ["Virtual li2"]),
    el("li", {class: "red-li"}, ["Virtual li3"])]),
  el("div", {class: "black-div"}, ["Virtual div"])
])

現(xiàn)在的div只是一個JS對象表示的DOM結(jié)構(gòu),頁面上并沒有這個結(jié)構(gòu),下面用來構(gòu)建真正的div

Element.prototype.render = function () {
    let el = document.createElement(this.tagName) //根據(jù)tagName構(gòu)建
    let props = this.props

    for (let propName in props) { // 設(shè)置節(jié)點的DOM屬性
        let propValue = props[propName]
        el.setAttribute(propName, propValue)
    }

    let children = this.children || []
    children.forEach(function (child) {
        let childEl = (child instanceof Element)
        ? child.render() // 如果子節(jié)點也是虛擬DOM,遞歸構(gòu)建DOM節(jié)點
        : document.createTextNode(child) // 如果字符串,只構(gòu)建文本節(jié)點
        el.appendChild(childEl)
    })
      return el
}

render方法會根據(jù)tagName構(gòu)建一個真正的DOM節(jié)點,然后設(shè)置這個節(jié)點的屬性,最后遞歸地把自己的子節(jié)點也構(gòu)建起來。所以只需要:

let divRoot = div.render()
document.body.appendChild(divRoot)

上面的運(yùn)行結(jié)果:

步驟二:比較兩棵虛擬DOM樹的差異(diff算法)

兩棵樹的完全差異比較的時間復(fù)雜度為O(n^3),這是不好的,又因為前端不會經(jīng)常進(jìn)行跨層地移動DOM元素,所以Virtual DOM只對同一層級的元素進(jìn)行比較,從而時間復(fù)雜度降為O(n)

深度優(yōu)先遍歷

在實際的代碼中,會對新舊兩棵樹進(jìn)行一個深度優(yōu)先的遍歷,這樣每個節(jié)點都會有一個唯一的標(biāo)記,在深度優(yōu)先遍歷的時候,每遍歷到一個節(jié)點就把改節(jié)點和新的數(shù)進(jìn)行對比,如果有差異就記錄到patches

// diff 函數(shù),對比兩棵樹
function diff (oldTree, newTree) {
  let index = 0 // 當(dāng)前節(jié)點的標(biāo)志
  let patches = {} // 用來記錄每個節(jié)點差異的對象
  dfsWalk(oldTree, newTree, index, patches)
  return patches
}

// 對兩棵樹進(jìn)行深度優(yōu)先遍歷
function dfsWalk (oldNode, newNode, index, patches) {
  // 對比oldNode和newNode的不同,記錄下來
  patches[index] = [...]

  diffChildren(oldNode.children, newNode.children, index, patches)
}

// 遍歷子節(jié)點
function diffChildren (oldChildren, newChildren, index, patches) {
  let leftNode = null
  let currentNodeIndex = index
  oldChildren.forEach(function (child, i) {
    let newChild = newChildren[i]
    currentNodeIndex = (leftNode && leftNode.count) // 計算節(jié)點的標(biāo)識
      ? currentNodeIndex + leftNode.count + 1
      : currentNodeIndex + 1
    dfsWalk(child, newChild, currentNodeIndex, patches) // 深度遍歷子節(jié)點
    leftNode = child
  })
}

例如,上面的div和新的div有差異,當(dāng)前的標(biāo)記是0,那么:

patches[0] = [{difference}, {difference}, ...] // 用數(shù)組存儲新舊節(jié)點的不同
四種差異

上面出現(xiàn)了四種新舊樹不同的情況:

REPLACE:節(jié)點類型變了,p變成了div,將舊節(jié)點卸載并裝載新節(jié)點

PROPS:不觸發(fā)節(jié)點的卸載和裝載,執(zhí)行節(jié)點的更新

TEXT:修改文本內(nèi)容

REORDER:移動、增加(多了li)、刪除節(jié)點,實際操作如圖:

所以我們定義了幾種差異類型:

let REPLACE = 0
patches[0] = [{
  type: REPALCE,
  node: newNode // el("div", props, children)  p換成div
}]

let PROPS = 1
patches[0] = [{
  type: REPALCE,
  node: newNode // el("p", props, children)
}, {
  type: PROPS,
  props: {//給p新增了id為container
    id: "container"
  }
}]

let TEXT = 2
patches[1] = [{//修改文本節(jié)點
  type: TEXT,
  content: "Virtual DOM2"
}]

let REORDER = 3  //重排見王沛的https://www.infoq.cn/article/react-dom-diff

最終Diff出來的結(jié)果類型如下:

{
    1: [ {type: REPLACE, node: Element} ],
    4: [ {type: TEXT, content: "after update"} ],
    5: [ {type: PROPS, props: {class: "marginLeft10"}}, {type: REORDER, moves: [{index: 2, type: 0}]} ],
    6: [ {type: REORDER, moves: [{index: 2, type: 0}]} ],
    8: [ {type: REORDER, moves: [{index: 2, type: 0}]} ],
    9: [ {type: TEXT, content: "Item 3"} ],
}
步驟三:把差異應(yīng)用到真正的DOM樹上

因為步驟一所構(gòu)建的 JavaScript 對象樹和render出來真正的DOM樹的信息、結(jié)構(gòu)是一樣的。所以我們可以對那棵DOM樹也進(jìn)行深度優(yōu)先的遍歷,遍歷的時候從步驟二生成的patches對象中找出當(dāng)前遍歷的節(jié)點差異,然后進(jìn)行 DOM 操作。

function patch (node, patches) {
  let walker = {index: 0}
  dfsWalk(node, walker, patches)
}

function dfsWalk (node, walker, patches) {
  let currentPatches = patches[walker.index] // 從patches拿出當(dāng)前節(jié)點的差異

  let len = node.childNodes
    ? node.childNodes.length
    : 0
  for (let i = 0; i < len; i++) { // 深度遍歷子節(jié)點
    let child = node.childNodes[i]
    walker.index++
    dfsWalk(child, walker, patches)
  }

  if (currentPatches) {
    applyPatches(node, currentPatches) // 對當(dāng)前節(jié)點進(jìn)行DOM操作
  }
}

applyPatches,根據(jù)不同類型的差異對當(dāng)前節(jié)點進(jìn)行 DOM 操作:

function applyPatches (node, currentPatches) {
  currentPatches.forEach(function (currentPatch) {
    switch (currentPatch.type) {
      case REPLACE:
        node.parentNode.replaceChild(currentPatch.node.render(), node)
        break
      case REORDER:
        reorderChildren(node, currentPatch.moves)
        break
      case PROPS:
        setProps(node, currentPatch.props)
        break
      case TEXT:
        node.textContent = currentPatch.content
        break
      default:
        throw new Error("Unknown patch type " + currentPatch.type)
    }
  })
}
結(jié)語

Virtual DOM 算法主要是實現(xiàn)上面步驟的三個函數(shù):element,diff,patch。然后就可以實際的進(jìn)行使用:

// 1. 構(gòu)建虛擬DOM
let tree = el("div", {"id": "container"}, [
    el("h1", {style: "color: blue"}, ["simple virtal dom"]),
    el("p", ["Hello, virtual-dom"]),
    el("ul", [el("li")])
])

// 2. 通過虛擬DOM構(gòu)建真正的DOM
let root = tree.render()
document.body.appendChild(root)

// 3. 生成新的虛擬DOM
let newTree = el("div", {"id": "container"}, [
    el("h1", {style: "color: red"}, ["simple virtal dom"]),
    el("p", ["Hello, virtual-dom"]),
    el("ul", [el("li"), el("li")])
])

// 4. 比較兩棵虛擬DOM樹的不同
let patches = diff(tree, newTree)

// 5. 在真正的DOM元素上應(yīng)用變更
patch(root, patches)

原理加1,頭發(fā)減一堆

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

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/102374.html

相關(guān)文章

  • 一起理解 Virtual DOM

    摘要:具體而言,就是每次數(shù)據(jù)發(fā)生變化,就重新執(zhí)行一次整體渲染。而給出了解決方案,就是。由于只關(guān)注,通過閱讀兩個庫的源碼,對于的定位有了更深一步的理解。第二個而且,技術(shù)本身不是目的,能夠更好地解決問題才是王道嘛。 前言 React 好像已經(jīng)火了很久很久,以致于我們對于 Virtual DOM 這個詞都已經(jīng)很熟悉了,網(wǎng)上也有非常多的介紹 React、Virtual DOM 的文章。但是直到前不久...

    Tangpj 評論0 收藏0
  • React Virtual DOM 理解

    摘要:二原理每個都有兩個,一個是新的,一個是原來的。三實現(xiàn)過程四算法的理解與實現(xiàn)本質(zhì)上就是在和之間做了一個緩存。將差異的應(yīng)用到真正的樹上對真實上的樹進(jìn)行深度優(yōu)先遍歷,在所有的差異列表中找出當(dāng)前遍歷的節(jié)點差異,然后根據(jù)不同進(jìn)行操作。 React Virtual DOM 一、概念 在react中,對于每個DOM對象都有一個相應(yīng)的虛擬DOM對象,相當(dāng)于DOM對象的輕量級副本 由于是Virtual...

    smallStone 評論0 收藏0
  • 從零開始,手寫一個簡易的Virtual DOM

    摘要:本文為筆者通過實際操作,實現(xiàn)了一個非常簡單的,加深對現(xiàn)今主流前端框架中的理解。用對象表示樹是用對象表示,并存儲在內(nèi)存中的。如果類型不一致,那么屬性一定是被更新的。如果有不相等的屬性,則認(rèn)為發(fā)生改變,需要處理的變化。 眾所周知,對前端而言,直接操作 DOM 是一件及其耗費(fèi)性能的事情,以 React 和 Vue 為代表的眾多框架普遍采用 Virtual DOM 來解決如今愈發(fā)復(fù)雜 Web ...

    forrest23 評論0 收藏0
  • 你不知道的Virtual DOM(一):Virtual Dom介紹

    摘要:不同的框架對這三個屬性的命名會有點差別,但表達(dá)的意思是一致的。它們分別是標(biāo)簽名屬性和子元素對象。我們先來看下頁面的更新一般會經(jīng)過幾個階段。元素有可能是數(shù)組的形式,需要將數(shù)組解構(gòu)一層。 歡迎關(guān)注我的公眾號睿Talk,獲取我最新的文章:showImg(https://segmentfault.com/img/bVbmYjo); 一、前言 目前最流行的兩大前端框架,React和Vue,都不約...

    lavor 評論0 收藏0
  • 構(gòu)建一個使用 Virtual-DOM 的前端模版引擎

    摘要:目錄前言問題的提出模板引擎和結(jié)合的實現(xiàn)編譯原理相關(guān)模版引擎的詞法分析語法分析與抽象語法樹代碼生成完整的結(jié)語前言本文嘗試構(gòu)建一個前端模板引擎,并且把這個引擎和進(jìn)行結(jié)合。于是就構(gòu)思了一個方案,在前端模板引擎上做手腳。 作者:戴嘉華 轉(zhuǎn)載請注明出處并保留原文鏈接( https://github.com/livoras/blog/issues/14 )和作者信息。 目錄 前言 問題的提出...

    imccl 評論0 收藏0
  • 你不知道的Virtual DOM(二):Virtual Dom的更新

    摘要:變化的只有種更新和刪除。頁面的元素的數(shù)量隨著而變。四總結(jié)本文詳細(xì)介紹如何實現(xiàn)一個簡單的算法,再根據(jù)計算出的差異去更新真實的。 歡迎關(guān)注我的公眾號睿Talk,獲取我最新的文章:showImg(https://segmentfault.com/img/bVbmYjo); 一、前言 目前最流行的兩大前端框架,React 和 Vue,都不約而同的借助 Virtual DOM 技術(shù)提高頁面的渲染...

    testbird 評論0 收藏0

發(fā)表評論

0條評論

BakerJ

|高級講師

TA的文章

閱讀更多
最新活動
閱讀需要支付1元查看
<