国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > JavaScript > 正文

vue2.0的虛擬DOM渲染思路分析

2019-11-19 13:18:17
字體:
供稿:網(wǎng)友

1、為什么需要虛擬DOM

前面我們從零開始寫了一個簡單的類Vue框架(文章鏈接),其中的模板解析和渲染是通過Compile函數(shù)來完成的,采用了文檔碎片代替了直接對頁面中DOM元素的操作,在完成數(shù)據(jù)的更改后通過appendChild函數(shù)將真實的DOM插入到頁面。

雖然采用的是文檔碎片,但是操作的還是真實的DOM。

而我們知道操作DOM的代價是昂貴的,所以vue2.0采用了虛擬DOM來代替對真實DOM的操作,最后通過某種機(jī)制來完成對真實DOM的更新,渲染視圖。

所謂的虛擬DOM,其實就是 用JS來模擬DOM結(jié)構(gòu),把DOM的變化操作放在JS層來做,盡量減少對DOM的操作 (個人認(rèn)為主要是因為操作JS比操作DOM快了不知道多少倍,JS運(yùn)行效率高)。然后對比前后兩次的虛擬DOM的變化,只重新渲染變化了的部分,而沒有變化的部分則不會重新渲染。

比如我們有如下的DOM結(jié)構(gòu)。

<ul id="list">   <li class="item1">Item 1</li>   <li class="item2">Item 2</li></ul>

我們完全可以用JS對象模擬上面的DOM結(jié)構(gòu),模擬后就會變成下面的這種結(jié)構(gòu)。

var vdom = {  tag: 'ul',  attr: {    id: 'list',  },  children: [    {      tag: 'li',      attrs: {        className: 'item',        children: ['Item 1']      },    },    {      tag: 'li',      attrs: {        className: 'item',        children: ['Item 2']      }    }  ]}

必須要注意一點的是:JS模擬的DOM結(jié)構(gòu)并沒有模擬所有DOM節(jié)點上的屬性、方法(因為DOM節(jié)點本身的屬性非常多,這也是DOM操作耗性能的一個點),而是只模擬了一部分和數(shù)據(jù)操作相關(guān)的屬性和方法。

2、怎么使用虛擬DOM

Vue 在 2.0 版本引入了 vdom 。其 vdom 是基于 snabbdom 庫所做的修改。snabbdom是一個開源的vdom庫。

snabbdom的主要作用就是 將傳入的JS模擬的DOM結(jié)構(gòu)轉(zhuǎn)換成虛擬的DOM節(jié)點。

先通過其中的h函數(shù)將JS模擬的DOM結(jié)構(gòu) 轉(zhuǎn)換成虛擬DOM之后,再通過其中的patch函數(shù)將虛擬DOM轉(zhuǎn)換成真實的DOM渲染到頁面中。

為了保證頁面的最小化渲染,snabbdom引入了Diff算法 ,通過Diff算法找出前后兩個虛擬DOM之間的差異,只更新改變了的DOM節(jié)點,而不重新渲染為改變的DOM節(jié)點。

在這里我不打算分析snabbdom的源碼來解釋到底snabbdom是怎么干成這件事的(主要是現(xiàn)階段沒到那個水平,哈哈。再者已經(jīng)有很多同學(xué)做過類似的分析,相關(guān)鏈接附在文章末尾)。

我會從snabbdom的使用角度來看Vue中的虛擬DOM是如何完成視圖渲染的。

我們先看一下snabbdom中兩個核心API的功能。

h()函數(shù):將傳入的JS模擬的DOM結(jié)構(gòu)模板轉(zhuǎn)換成vnode。(vnode是一個純JS對象)
patch()函數(shù):將虛擬的DOM節(jié)點渲染到頁面中。

我們提供一個實例來看一下snabbdom的實際作用。

<!DOCTYPE html><html lang="en"><head>  <meta charset="UTF-8">  <meta name="viewport" content="width=device-width, initial-scale=1.0">  <meta http-equiv="X-UA-Compatible" content="ie=edge">  <title>Document</title></head><body>  <div id="container"></div>  <button id="btn-change">change</button>  <!-- 引入snabbdom庫,先不必糾結(jié)為什么這樣引入,以及每個文件的作用。本篇文章只是介紹一下虛擬DOM的工作方式,并不涉及原理解析  主要是因為博主目前功力尚淺,有興趣的小伙伴可以另行研究 -->  <script src="https://cdn.bootcss.com/snabbdom/0.7.1/snabbdom.js"></script>  <script src="https://cdn.bootcss.com/snabbdom/0.7.1/snabbdom-class.js"></script>  <script src="https://cdn.bootcss.com/snabbdom/0.7.1/snabbdom-props.js"></script>  <script src="https://cdn.bootcss.com/snabbdom/0.7.1/snabbdom-style.js"></script>  <script src="https://cdn.bootcss.com/snabbdom/0.7.1/snabbdom-eventlisteners.js"></script>  <script src="https://cdn.bootcss.com/snabbdom/0.7.1/h.js"></script>  <script>    //定義patch函數(shù)    var patch = snabbdom.init([      snabbdom_class,      snabbdom_props,      snabbdom_style,      snabbdom_eventlisteners    ])    //定義h函數(shù)    var h = snabbdom.h;    //生成一個vnode    var vnode = h('ul#list',{},[      h('li.item',{},['Item 1']),      h('li.item',{},['Item 2']),    ])     //console.log(vnode);    //獲取container    var container = document.getElementById('container');    patch(container,vnode);//初次渲染    var btn = document.getElementById('btn-change');    btn.onclick = function() {      var newVnode = h('ul#list',{},[        h('li.item',{},['Item 1']),        h('li.item',{},['Item B']),        h('li.item',{},['Item 3']),      ])      patch(vnode,newVnode);//再次渲染       vnode = newVnode;//將修改后的newVnode賦值給vnode          }  </script></body></html>

思路分析:

  • 我們先通過h函數(shù)創(chuàng)建一個虛擬DOM節(jié)點,通過patch函數(shù)將虛擬DOM渲染到頁面。
  • 點擊btn按鈕時,更新ul#list列表的數(shù)據(jù),改變了第二個li元素的值并且新增了一個li元素,第一個li元素的值并沒有改變。我們再次通過patch函數(shù)將更新后的數(shù)據(jù)渲染到頁面上。可以看到只有第二個和第三個li發(fā)生了更新,而第一個li由于沒有改變,并沒有重新渲染。

vue中的模板解析和渲染的核心就是:通過類似snabbdom的h()和patch()的函數(shù),先將模板解析成vnode,如果是初次渲染,則通過patch(container,vnode)將vnode渲染至頁面,如果是二次渲染,則通過patch(vnode,newVnode),先通過Diff算法比較原vnode和newVnode的差異,以最小的代價重新渲染頁面。

總結(jié)

以上所述是小編給大家介紹的vue2.0的虛擬DOM渲染思路分析,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復(fù)大家的。在此也非常感謝大家對武林網(wǎng)網(wǎng)站的支持!

發(fā)表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發(fā)表
主站蜘蛛池模板: 古蔺县| 惠水县| 晋城| 中宁县| 达州市| 怀宁县| 台山市| 曲阳县| 收藏| 赣榆县| 莱西市| 商水县| 贵溪市| 嵊州市| 遵义县| 桐梓县| 镇宁| 华蓥市| 清原| 政和县| 合江县| 雷波县| 察哈| 疏附县| 西丰县| 南昌市| 鹤岗市| 石台县| 徐水县| 象山县| 沙坪坝区| 彭州市| 怀远县| 内丘县| 项城市| 临泽县| 高碑店市| 增城市| 清远市| 宜章县| 郁南县|