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

資訊專欄INFORMATION COLUMN

從Preact了解一個類React的框架是怎么實(shí)現(xiàn)的(二): 元素diff

張巨偉 / 3022人閱讀

摘要:本系列文章將重點(diǎn)分析類似于的這類框架是如何實(shí)現(xiàn)的,歡迎大家關(guān)注和討論。作為一個極度精簡的庫,函數(shù)是屬于本身的。

前言

  首先歡迎大家關(guān)注我的掘金賬號和Github博客,也算是對我的一點(diǎn)鼓勵,畢竟寫東西沒法獲得變現(xiàn),能堅(jiān)持下去也是靠的是自己的熱情和大家的鼓勵。
  之前分享過幾篇關(guān)于React的文章:

React技術(shù)內(nèi)幕: key帶來了什么

React技術(shù)內(nèi)幕: setState的秘密

  其實(shí)我在閱讀React源碼的時候,真的非常痛苦。React的代碼及其復(fù)雜、龐大,閱讀起來挑戰(zhàn)非常大,但是這卻又擋不住我們的React的原理的好奇。前段時間有人就安利過Preact,千行代碼就基本實(shí)現(xiàn)了React的絕大部分功能,相比于React動輒幾萬行的代碼,Preact顯得別樣的簡潔,這也就為了我們學(xué)習(xí)React開辟了另一條路。本系列文章將重點(diǎn)分析類似于React的這類框架是如何實(shí)現(xiàn)的,歡迎大家關(guān)注和討論。如有不準(zhǔn)確的地方,歡迎大家指正。
  
  在上篇文章從preact了解一個類React的框架是怎么實(shí)現(xiàn)的(一): 元素創(chuàng)建我們了解了我們平時所書寫的JSX是怎樣轉(zhuǎn)化成Preact中的虛擬DOM結(jié)構(gòu)的,接下來我們就要了解一下這些虛擬DOM節(jié)點(diǎn)是如何渲染成真實(shí)的DOM節(jié)點(diǎn)的以及虛擬DOM節(jié)點(diǎn)的改變?nèi)绾斡成涞秸鎸?shí)DOM節(jié)點(diǎn)的改變(也就是diff算法的過程)。這篇文章相比第一篇會比較冗長和枯燥,為了能集中分析diff過程,我們只關(guān)注dom元素,暫時不去考慮組件。
  

渲染與diff render函數(shù)

  我們知道在React中渲染是并不是由React完成的,而是由ReactDOM中的render函數(shù)去實(shí)現(xiàn)的。其實(shí)在最早的版本中,render函數(shù)也是屬于React的,只不過后來React的開發(fā)者想實(shí)現(xiàn)一個于平臺無關(guān)的庫(其目的也是為了React Native服務(wù)的),因此將Web中渲染的部分獨(dú)立成ReactDOM庫。Preact作為一個極度精簡的庫,render函數(shù)是屬于Preact本身的。Preact的render函數(shù)與ReactDOM的render函數(shù)也是有有所區(qū)別的:

ReactDOM.render(
  element,
  container,
  [callback]
)

  ReactDOM.render接受三個參數(shù),element是需要渲染的React元素,而container掛載點(diǎn),即React元素將被渲染進(jìn)container中,第三個參數(shù)callback是可選的,當(dāng)組件被渲染或者更新的時候會被調(diào)用。ReactDOM.render會返回渲染組元素的真實(shí)DOM節(jié)點(diǎn)。如果之前container中含有dom節(jié)點(diǎn),則渲染時會將之前的所有節(jié)點(diǎn)清除。例如:

html:

Hello React!

javascript:

ReactDOM.render(
  

Hello, world!

, document.getElementById("root") );

  最終的顯示效果為:

Hello, world!

  而Preact的render函數(shù)為:
  

Preact.render(
  vnode, 
  parent, 
  [merge]
)

  Preact.renderReactDOM.render的前兩個參數(shù)代表的意義相同,區(qū)域在于最后一個,Preact.render可選的第三個參數(shù)merge,要求必須是第二個參數(shù)的子元素,是指會被替換的根節(jié)點(diǎn),否則,如果沒有這個參數(shù),Preact 默認(rèn)追加,而不是像React進(jìn)行替換。
  
  例如不存在第三個參數(shù)的情況下:

html:

Hello Preact!

javascript:

preact.render(
  

Hello, world!

, document.getElementById("root") );

  最終的顯示效果為:

Hello Preact
Hello, world!

  如果調(diào)用函數(shù)有第三個參數(shù):

javascript:

preact.render(
  

Hello, world!

, document.getElementById("root"), document.getElementById("container") );

  顯示效果是:

Hello, world!

  

實(shí)現(xiàn)

  其實(shí)在Preact中無論是初次渲染還是之后虛擬DOM改變導(dǎo)致的UI更新最終調(diào)用的都是diff函數(shù),這也是非常合理的,畢竟我們可以將首次渲染當(dāng)做是diff過程中用現(xiàn)有的虛擬dom去與空的真實(shí)dom基礎(chǔ)上進(jìn)行更新的過程。下面我們首先給出整個diff過程的大致流程圖,我們可以對照流程圖對代碼進(jìn)行分析:
  
  
  首先從render函數(shù)入手,render函數(shù)調(diào)用的就是diff函數(shù):

function render(vnode, parent, merge) {
    return diff(merge, vnode, {}, false, parent, false);
}

  我們可以看到Preact中的render調(diào)用了diff函數(shù),而diff定義在vdom/diff中:

function diff(dom, vnode, context, mountAll, parent, componentRoot) {

    // diffLevel為 0 時表示第一次進(jìn)入diff函數(shù)
    if (!diffLevel++) {
        // 第一次執(zhí)行diff,查看我們是否在diff SVG元素或者是元素在SVG內(nèi)部
        isSvgMode = parent!=null && parent.ownerSVGElement!==undefined;

        // hydration 指示的是被diff的現(xiàn)存元素是否含有屬性props的緩存
        // 屬性props的緩存被存在dom節(jié)點(diǎn)的__preactattr_屬性中
        hydrating = dom!=null && !(ATTR_KEY in dom);
    }

    let ret = idiff(dom, vnode, context, mountAll, componentRoot);

    // 如果父節(jié)點(diǎn)之前沒有創(chuàng)建的這個子節(jié)點(diǎn),則將子節(jié)點(diǎn)添加到父節(jié)點(diǎn)之后
    if (parent && ret.parentNode!==parent) parent.appendChild(ret);

    // diffLevel回減到0說明已經(jīng)要結(jié)束diff的調(diào)用
    if (!--diffLevel) {
        hydrating = false;
        // 負(fù)責(zé)觸發(fā)組件的componentDidMount生命周期函數(shù)
        if (!componentRoot) flushMounts();
    }

    return ret;
}

  這部分的函數(shù)內(nèi)容比較龐雜,很難做到面面俱到,我會在代碼中做相關(guān)的注釋。diff函數(shù)主要負(fù)責(zé)就是將當(dāng)前的虛擬node節(jié)點(diǎn)映射到真實(shí)的DOM節(jié)點(diǎn)中。參數(shù)如下:

vnode: 不用說,就是我們需要渲染的虛擬dom節(jié)點(diǎn)

parent: 就是你要將虛擬dom掛載的父節(jié)點(diǎn)

dom: 這里的dom其實(shí)就是當(dāng)前的vnode所對應(yīng)的之前未更新的真實(shí)dom。那么就有兩種可能: 第一就是null或者是上面例子的contaienr(就是render函數(shù)對應(yīng)的第三個參數(shù)),其本質(zhì)都是首次渲染,第二種就是vnode的對應(yīng)的未更新的真實(shí)dom,那么對應(yīng)的就是渲染刷新界面。

context: 組件相關(guān),暫時可以不考慮,對應(yīng)React中的context

mountAll: 組件相關(guān),暫時可以不考慮

componentRoot: 組件相關(guān),暫時可以不考慮

  vnode對應(yīng)的就是一個遞歸的結(jié)構(gòu),那么不用想diff函數(shù)肯定也是遞歸的。我們首先看一下函數(shù)初始的幾個變量:

diffLevel:用來記錄當(dāng)前渲染的層數(shù)(遞歸的深度),其實(shí)在代碼中并沒有在進(jìn)入每層遞歸的時候都增加并且退出遞歸的時候減小。只是記錄了是不是渲染的第一層,所以對應(yīng)的值只有01。

isSvgMode:用來指代當(dāng)前的渲染是否內(nèi)SVG元素的內(nèi)部或者我們是否在diff一個SVG元素(SVG元素需要特殊處理)。

hydrating: 這個變量是我一直所困惑的,我還專門查了一下,hydrating指的是保濕、吸水 的意思。hydrating = dom != null && !(ATTR_KEY in dom);(ATTR_KEY對應(yīng)常量__preactattr_,preact會將props等緩存信息存儲在dom的__preactattr_屬性中),作者給的是下面的注釋:

hydration is indicated by the existing element to be diffed not having a prop cache

也就是說hydrating是指當(dāng)前的diff的元素沒有緩存但是對應(yīng)的dom元素必須存在。那么什么時候才會出現(xiàn)dom節(jié)點(diǎn)中沒有存儲緩存?只有當(dāng)前的dom節(jié)點(diǎn)并不是由Preact所創(chuàng)建并渲染的才會使得hydrating為true。

  idiff函數(shù)就是diff算法的內(nèi)部實(shí)現(xiàn),相對來說代碼會比較復(fù)雜,idiff會返回虛擬dom對應(yīng)創(chuàng)建的真實(shí)dom節(jié)點(diǎn)。下面的代碼是是向父級元素有選擇性添加創(chuàng)建的dom節(jié)點(diǎn),之所以這么做,主要是有可能之前該節(jié)點(diǎn)就沒有渲染過,所以需要將新創(chuàng)建的dom節(jié)點(diǎn)添加到父級dom。但是如果僅僅只是修改了之前dom中的某一個屬性(比如樣式),那么其實(shí)是不需要添加的,因?yàn)樵揹om節(jié)點(diǎn)已經(jīng)存在于父級dom。
  
  后面的內(nèi)容,一方面結(jié)束遞歸之后,回置diffLevel(diffLevel此時應(yīng)該為0,表明此時要退出diff函數(shù)),退出diff前,將hydrating置為false,相當(dāng)于一個復(fù)位的功能。下面的flushMounts函數(shù)是組件相關(guān),在這里我們只需要知道它要做的就是去執(zhí)行所有剛才安裝組件的componentDidMount生命周期函數(shù)。
  
  下面讓我們看看idiff的實(shí)現(xiàn)(代碼已經(jīng)分塊,具體見注釋),代碼比較長,可以先大致瀏覽一下,做到心里有數(shù),下面會逐塊分析,可以對照流程圖看:

/** 內(nèi)部的diff函數(shù) */
function idiff(dom, vnode, context, mountAll, componentRoot) {
    // block-1
    let out = dom, prevSvgMode = isSvgMode;

    // 空的node 渲染空的文本節(jié)點(diǎn)
    if (vnode==null || typeof vnode==="boolean") vnode = "";

    // String & Number 類型的節(jié)點(diǎn) 創(chuàng)建/更新 文本節(jié)點(diǎn)
    if (typeof vnode==="string" || typeof vnode==="number") {

        // 更新如果存在的原有文本節(jié)點(diǎn)
        // 這里如果節(jié)點(diǎn)值是文本類型,其父節(jié)點(diǎn)又是文本類型的節(jié)點(diǎn),則直接更新
        if (dom && dom.splitText!==undefined && dom.parentNode && (!dom._component || componentRoot)) {
            if (dom.nodeValue!=vnode) {
                dom.nodeValue = vnode;
            }
        }
        else {
            // 不是文本節(jié)點(diǎn),替換之前的節(jié)點(diǎn),回收之前的節(jié)點(diǎn)
            out = document.createTextNode(vnode);
            if (dom) {
                if (dom.parentNode) dom.parentNode.replaceChild(out, dom);
                recollectNodeTree(dom, true);
            }
        }

        out[ATTR_KEY] = true;
        return out;
    }

    // block-2
    // 如果是VNode代表的是一個組件,使用組件的diff
    let vnodeName = vnode.nodeName;
    if (typeof vnodeName==="function") {
        return buildComponentFromVNode(dom, vnode, context, mountAll);
    }

    // block-3    
    // 沿著樹向下時記錄記錄存在的SVG命名空間
    isSvgMode = vnodeName==="svg" ? true : vnodeName==="foreignObject" ? false : isSvgMode;

    // 如果不是一個已經(jīng)存在的元素或者類型有問題,則重新創(chuàng)建一個
    vnodeName = String(vnodeName);
    if (!dom || !isNamedNode(dom, vnodeName)) {
        out = createNode(vnodeName, isSvgMode);

        if (dom) {
            // 移動dom中的子元素到out中
            while (dom.firstChild) out.appendChild(dom.firstChild);

            // 如果之前的元素已經(jīng)屬于某一個DOM節(jié)點(diǎn),則將其替換
            if (dom.parentNode) dom.parentNode.replaceChild(out, dom);

            // 回收之前的dom元素(跳過非元素類型)
            recollectNodeTree(dom, true);
        }
    }

    // block-4
    let fc = out.firstChild,
        props = out[ATTR_KEY],
        vchildren = vnode.children;

    if (props==null) {
        props = out[ATTR_KEY] = {};
        for (let a=out.attributes, i=a.length; i--; ) props[a[i].name] = a[i].value;
    }

    // 優(yōu)化: 對于元素只包含一個單一文本節(jié)點(diǎn)的優(yōu)化路徑
    if (!hydrating && vchildren && vchildren.length===1 && typeof vchildren[0]==="string" && fc!=null && fc.splitText!==undefined && fc.nextSibling==null) {
        if (fc.nodeValue!=vchildren[0]) {
            fc.nodeValue = vchildren[0];
        }
    }
    // 否則,如果有存在的子節(jié)點(diǎn)或者新的孩子節(jié)點(diǎn),執(zhí)行diff
    else if (vchildren && vchildren.length || fc!=null) {
        innerDiffNode(out, vchildren, context, mountAll, hydrating || props.dangerouslySetInnerHTML!=null);
    }

    // 將props和atrributes從VNode中應(yīng)用到DOM元素
    diffAttributes(out, vnode.attributes, props);

    // 恢復(fù)之前的SVG模式
    isSvgMode = prevSvgMode;

    return out;
}

  idiff函數(shù)所接受的參數(shù)與diff是完全相同的,但是二者也是有所區(qū)別的。diff在渲染過程(或者更新過程)中僅僅會調(diào)用一次,所以說diff函數(shù)接受的vnode就是整個應(yīng)用的虛擬dom,而dom也就是當(dāng)前整個虛擬dom所對應(yīng)的節(jié)點(diǎn)。但是idiff的調(diào)用是遞歸的,因此domvnode開始時diff函數(shù)相等,但是在之后遞歸的過程中,就對應(yīng)的是整個應(yīng)用的部分。

首先來看第一塊(block-1)的代碼:

  變量prevSvgMode用來存儲之前的isSvgMode,目的就是在退出這一次遞歸調(diào)用時恢復(fù)到調(diào)用前的值。然后如果vnode是null或者布爾類型,都按照空字符去處理。接下的渲染是整對于字符串(sting或者number類型),主要分為兩部分: 更新或者創(chuàng)建元素。如果dom本身存在并且就是一個文本節(jié)點(diǎn),那就只需要將其中的值更新為當(dāng)前的值即可。否則創(chuàng)建一個新的文本節(jié)點(diǎn),并且將其替換到父元素上,并回收之前的節(jié)點(diǎn)值。因?yàn)槲谋竟?jié)點(diǎn)是沒有什么需要緩存的屬性值(文本的顏色等屬性實(shí)際是存儲的父級的元素中),所以直接將其ATTR_KEY(實(shí)際值為__preactattr_)賦值為true,并返回新創(chuàng)建的元素。這段代碼有兩個需要注意的地方:

if (dom.nodeValue!=vnode) {
    dom.nodeValue = vnode;
}

  為什么在賦值文本節(jié)點(diǎn)值時,需要首先進(jìn)行一個判斷?根據(jù)代碼注釋得知Firfox瀏覽器不會默認(rèn)做等值比較(其他的瀏覽器例如Chrome即使直接賦值,如果相等也不會修改dom元素),所以人為的增加了比較的過程,目的就是為了防止文本節(jié)點(diǎn)每次都會被更新,這算是一個瀏覽器怪癖(quirk)。

  回收dom節(jié)點(diǎn)的recollectNodeTree函數(shù)做了什么?看代碼:

/**
 * 遞歸地回收(或者卸載)節(jié)點(diǎn)及其后代節(jié)點(diǎn)
 * @param node
 * @param unmountOnly 如果為`true`,僅僅觸發(fā)卸載的生命周期,跳過刪除
 */
function recollectNodeTree(node, unmountOnly) {
    let component = node._component;
    if (component) {
        // 如果該節(jié)點(diǎn)屬于某個組件,卸載該組件(最終在這里遞歸),主要包括組件的回收和相依卸載生命周期的調(diào)用
        unmountComponent(component);
    }
    else {
        // 如果節(jié)點(diǎn)含有ref函數(shù),則執(zhí)行ref函數(shù),參數(shù)為null(這里是React的規(guī)范,用于取消設(shè)置引用)
        // 確實(shí)在React如果設(shè)置了ref的話,在卸載的時候,也會被回調(diào),得到的參數(shù)是null
        if (node[ATTR_KEY]!=null && node[ATTR_KEY].ref) node[ATTR_KEY].ref(null);

        if (unmountOnly===false || node[ATTR_KEY]==null) {
            //要做的無非是從父節(jié)點(diǎn)將該子節(jié)點(diǎn)刪除
            removeNode(node);
        }

        //遞歸刪除子節(jié)點(diǎn)
        removeChildren(node);
    }
}
/**
 * 回收/卸載所有的子元素
 * 我們在這里使用了.lastChild而不是使用.firstChild,是因?yàn)樵L問節(jié)點(diǎn)的代價更低。
 */
export function removeChildren(node) {
    node = node.lastChild;
    while (node) {
        let next = node.previousSibling;
        recollectNodeTree(node, true);
        node = next;
    }
}
/** 從父節(jié)點(diǎn)刪除該節(jié)點(diǎn)
 *    @param {Element} node        待刪除的節(jié)點(diǎn)
 */
function removeNode(node) {
    let parentNode = node.parentNode;
    if (parentNode) parentNode.removeChild(node);
}

  我們看到在函數(shù)recollectNodeTree中,如果dom元素屬于某個組件,首先遞歸卸載組件(不是本次講述的重點(diǎn),主要包括組件的回收和相依卸載生命周期的調(diào)用)。否則,只需要先判別該dom節(jié)點(diǎn)中是否被在jsx中存在ref函數(shù)(也是緩存在__preactattr_屬性中),因?yàn)榇嬖?b>ref函數(shù)時,我們在組件卸載時以null參數(shù)作為回調(diào)(React文檔做了相應(yīng)的規(guī)定,詳情見Refs and the DOM)。recollectNodeTree中第二個參數(shù)unmountOnly,表示僅僅觸發(fā)卸載的生命周期,跳過刪除的過程,如果unmountOnlyfalse或者dom中的ATTR_KEY屬性不存在(說明這個屬性不是preact所渲染的,否則肯定會存在該屬性),則直接將其從父節(jié)點(diǎn)刪除。最后遞歸刪除子節(jié)點(diǎn),我們可以看到遞歸刪除子元素的過程是從右到左刪除的(首先刪除的lastChild元素),主要考慮到的是從后訪問會有性能的優(yōu)勢。我們在這里(block-1)調(diào)用函數(shù)recollectNodeTree的第二個參數(shù)是true,原因是在調(diào)用之前我們已經(jīng)將其在父元素中進(jìn)行替換,所以是不需要進(jìn)行調(diào)用的函數(shù)removeNode再進(jìn)行刪除該節(jié)點(diǎn)的。
 

第二塊代碼,主要是針對的組件的渲染,如果vnode.nodeName對應(yīng)的是函數(shù)類型,表明要渲染的是一個組件,直接調(diào)用了函數(shù)buildComponentFromVNode(組件不是本次敘述內(nèi)容)。

第三塊代碼,首先:

isSvgMode = vnodeName==="svg" ? true : vnodeName==="foreignObject" ? false : isSvgMode;

  變量isSvgMode還是用來標(biāo)記當(dāng)前創(chuàng)建的元素是否是SVG元素。foreignObject元素允許包含外來的XML命名空間,一個foreignObject內(nèi)部的任何SVG元素都不會被繪制,所以如果是vnodeNameforeignObject話,isSvgMode會被置為false(其實(shí)Svg對我來說也是比較生疏的內(nèi)容,但是不影響我們分析整個渲染過程)。

    // 如果不是一個已經(jīng)存在的元素或者類型有問題,則重新創(chuàng)建一個
    vnodeName = String(vnodeName);
    if (!dom || !isNamedNode(dom, vnodeName)) {
        out = createNode(vnodeName, isSvgMode);

        if (dom) {
            // 移動dom中的子元素到out中
            while (dom.firstChild) out.appendChild(dom.firstChild);

            // 如果之前的元素已經(jīng)屬于某一個DOM節(jié)點(diǎn),則將其替換
            if (dom.parentNode) dom.parentNode.replaceChild(out, dom);

            // 回收之前的dom元素(跳過非元素類型)
            recollectNodeTree(dom, true);
        }
    }

  然后開始嘗試創(chuàng)建dom元素,如果之前的dom為空(說明之前沒有渲染)或者dom的名稱與vnode.nodename不一致時,說明我們要創(chuàng)建新的元素,然后如果之前的dom節(jié)點(diǎn)中存在子元素,則將其全部移入新創(chuàng)建的元素中。如果之前的dom已經(jīng)有父元素了,則將其替換成新的元素,最后回收該元素。
  在判斷節(jié)點(diǎn)dom類型與虛擬dom的vnodeName類型是否相同時使用了函數(shù)isNamedNode:
  

function isNamedNode(node, nodeName) {
    return node.normalizedNodeName===nodeName || node.nodeName.toLowerCase()===nodeName.toLowerCase();
}

  如果節(jié)點(diǎn)是由Preact創(chuàng)建的(即由函數(shù)createNode創(chuàng)建的),其中dom節(jié)點(diǎn)中含有屬性normalizedNodeName(node.normalizedNodeName = nodeName),則使用normalizedNodeName去判斷節(jié)點(diǎn)類型是否相等,否則直接采用dom節(jié)點(diǎn)中的nodeName屬性去判斷。
 
  到此為止渲染的當(dāng)前虛擬dom的過程已經(jīng)結(jié)束,接下來就是處理子元素的過程。

第四塊代碼:

    let fc = out.firstChild,
        props = out[ATTR_KEY],
        vchildren = vnode.children;

    if (props==null) {
        props = out[ATTR_KEY] = {};
        for (let a=out.attributes, i=a.length; i--; ) props[a[i].name] = a[i].value;
    }

    // 優(yōu)化: 對于元素只包含一個單一文本節(jié)點(diǎn)的優(yōu)化路徑
    if (!hydrating && vchildren && vchildren.length===1 && typeof vchildren[0]==="string" && fc!=null && fc.splitText!==undefined && fc.nextSibling==null) {
        if (fc.nodeValue!=vchildren[0]) {
            fc.nodeValue = vchildren[0];
        }
    }
    // 否則,如果有存在的子節(jié)點(diǎn)或者新的孩子節(jié)點(diǎn),執(zhí)行diff
    else if (vchildren && vchildren.length || fc!=null) {
        innerDiffNode(out, vchildren, context, mountAll, hydrating || props.dangerouslySetInnerHTML!=null);
    }

  然后我們看到,如果out是新創(chuàng)建的元素或者該元素不是由Preact創(chuàng)建的(即不存在屬性__preactattr_),我們會初始化out中的__preactattr_屬性中并將out元素(剛創(chuàng)建的dom元素)中屬性attributes緩存在out元素的ATTR_KEY(__preactattr_)屬性上。但是需要注意的是,比如某個節(jié)點(diǎn)的屬性發(fā)生改變,比如name1變成了2,那么out屬性中的緩存(__preactattr_)也需要得到更新,但是更新的操作并不發(fā)生在這里,而是下面的diffAttributes函數(shù)中。
  
  接下來就是處理子元素只有一個文本節(jié)點(diǎn)的情況(其實(shí)這部分也可以沒有,通過下一層的遞歸也能解決,這樣做只不過是為了優(yōu)化性能),比如處理下面的情形:

1

  進(jìn)入單個節(jié)點(diǎn)的判斷條件也是比較明確的,唯一需要注意的一點(diǎn)是,必須滿足hydrating不為true,因?yàn)槲覀冎喇?dāng)hydratingtrue是說明當(dāng)前的節(jié)點(diǎn)并不是由Preact渲染的,因此不能進(jìn)行直接的優(yōu)化,需要由下一層遞歸中創(chuàng)建新的文本元素。
  

    //將props和atrributes從VNode中應(yīng)用到DOM元素
    diffAttributes(out, vnode.attributes, props);
    // 恢復(fù)之前的SVG模式
    isSvgMode = prevSvgMode;
    return out;

  函數(shù)diffAttributes的主要作用就是將虛擬dom中attributes更新到真實(shí)的dom中(后面詳細(xì)講)。最后重置變量isSvgMode,并返回vnode所渲染的真實(shí)dom節(jié)點(diǎn)。
  
  看完了函數(shù)idiff,接下來要關(guān)心的就是,在idiff中對虛擬dom的子元素調(diào)用的innerDiffNode函數(shù)(代碼依然很長,我們依然做分塊,對照流程圖看):

function innerDiffNode(dom, vchildren, context, mountAll, isHydrating) {
    let originalChildren = dom.childNodes,
        children = [],
        keyed = {},
        keyedLen = 0,
        min = 0,
        len = originalChildren.length,
        childrenLen = 0,
        vlen = vchildren ? vchildren.length : 0,
        j, c, f, vchild, child;

    // block-1
    // 創(chuàng)建一個包含key的子元素和一個不包含有子元素的Map
    if (len!==0) {
        for (let i=0; i

  首先看innerDiffNode函數(shù)的參數(shù):

dom: diff的虛擬子元素的父元素對應(yīng)的真實(shí)dom節(jié)點(diǎn)

vchildren: diff的虛擬子元素

context: 類似于React中的context,組件使用

mountAll: 組件相關(guān),暫時可以不考慮

componentRoot: 組件相關(guān),暫時可以不考慮

  函數(shù)代碼將近百行,為了方便閱讀,我們將其分為四個部分(看代碼注釋):

第一部分代碼:

// 創(chuàng)建一個包含key的子元素和一個不包含有子元素的Map
if (len!==0) {
    //len === dom.childNodes.length
    for (let i=0; i

  我們所希望的diff的過程肯定是以最少的dom操作使得更改后的dom與虛擬dom相匹配,所以之前父節(jié)點(diǎn)的dom重用也是非常必要。len是父級dom的子元素個數(shù),首先對所有的子元素進(jìn)行遍歷,如果該元素是由Preact所渲染(也就是有props的緩存)并且含有key值(不考慮組件的情況下,我們暫時只看該元素props中是否有key值),我們將其存儲在keyed中,否則如果該元素也是Preact所渲染(有props的緩存)或者滿足條件(child.splitText!==undefined ? (isHydrating ? child.nodeValue.trim() : true) : isHydrating)時,我們將其分配到children中。這樣我們其實(shí)就將子元素劃分為兩類,一類是帶有key值的子元素,一類是沒有key的子元素。

  關(guān)于條件(child.splitText!==undefined ? (isHydrating ? child.nodeValue.trim() : true) : isHydrating)我們分析一下,我們知道hydratingtrue時表示的是dom元素不是Preact創(chuàng)建的,我們知道調(diào)用函數(shù)innerDiffNode時,isHydrating的值是hydrating || props.dangerouslySetInnerHTML!=null,那么isHydratingtrue表示的就是子dom節(jié)點(diǎn)不是由Preact所創(chuàng)建的,那么現(xiàn)在看起來上面的判斷條件也非常容易理解了。如果節(jié)點(diǎn)child不是文本節(jié)點(diǎn),根據(jù)該節(jié)點(diǎn)是否是由Preact所創(chuàng)建的做決定,如果是不是由Preact創(chuàng)建的,則添加到children,否則不添加。如果是文本節(jié)點(diǎn)的話,如果是由Preact創(chuàng)建的話則添加,否則執(zhí)行child.nodeValue.trim(),我們知道函數(shù)trim返回的是去掉字符串前后空格的新字符串,如果該節(jié)點(diǎn)有非空字符,則會被添加到children中,否則不添加。這樣做的目的也無非是最大程度利用之前的文本節(jié)點(diǎn),減少創(chuàng)建不必要的文本節(jié)點(diǎn)。

第二部分代碼:

if (vlen!==0) {

    for (let i=0; i

  該部分代碼首先對虛擬dom中的子元素進(jìn)行遍歷,對每一個子元素,首先判斷該子元素是否含有屬性key,如果含有則在keyed中查找對應(yīng)keyed的dom元素,并在keyed將該元素刪除。否則在children查找是否含有和該元素相同類型的節(jié)點(diǎn)(利用函數(shù)isSameNodeType),如果查找到相同類型的節(jié)點(diǎn),則在children中刪除并根據(jù)對應(yīng)的情況(即查到的元素在children查找范圍的首尾)縮小排查范圍。然后遞歸執(zhí)行函數(shù)idiff,如果之前child沒有查找到的話,會在idiff中創(chuàng)建對應(yīng)類型的節(jié)點(diǎn)。然后根據(jù)之前的所分析的,idiff會返回新的dom節(jié)點(diǎn)。
  
  如果idiff返回dom不為空并且該dom與原始dom中對應(yīng)位置的dom不相同時,將其添加到父節(jié)點(diǎn)。如果不存在對應(yīng)位置的真實(shí)節(jié)點(diǎn),則直接添加到父節(jié)點(diǎn)。如果child已經(jīng)添加到對應(yīng)位置的真實(shí)dom后,則直接將其移除當(dāng)前位置的真實(shí)dom,否則都將其添加到對應(yīng)位置之前。

第三塊代碼:

    if (keyedLen) {
        for (let i in keyed) if (keyed[i]!==undefined) recollectNodeTree(keyed[i], false);
    }
    // 移除沒有父節(jié)點(diǎn)的不帶有key值的子元素
    while (min<=childrenLen) {
        if ((child = children[childrenLen--])!==undefined) recollectNodeTree(child, false);
    }

  這段代碼所作的工作就是將keyed中與children中沒有用到的原始dom節(jié)點(diǎn)回收。到此我們已經(jīng)基本講完了整個diff的所有大致流程,還剩idiff中的diffAttributes函數(shù)沒有講,因?yàn)槔锩嫔婕暗絛om中的事件觸發(fā),所以還是有必要講一下:
  

function diffAttributes(dom, attrs, old) {
    let name;

    // 通過將其設(shè)置為undefined,移除不在vnode中的屬性
    for (name in old) {
        // 判斷的條件是如果old[name]中存在,但attrs[name]不存在
        if (!(attrs && attrs[name]!=null) && old[name]!=null) {
            setAccessor(dom, name, old[name], old[name] = undefined, isSvgMode);
        }
    }
    // 增加或者更新的屬性
    for (name in attrs) {
        // 如果attrs中的屬性不是 children或者 innerHTML 并且
        // 要么 之前的old里面沒有該屬性 ====> 說明是新增屬性
        // 要么 如果name是value或者checked屬性(表單), attrs[name] 與 dom[name] 不同,或者不是value或者checked屬性,則和old[name]屬性不同 ====> 說明是更新屬性
        if (name!=="children" && name!=="innerHTML" && (!(name in old) || attrs[name]!==(name==="value" || name==="checked" ? dom[name] : old[name]))) {
            setAccessor(dom, name, old[name], old[name] = attrs[name], isSvgMode);
        }
    }
}

  diffAttributes的參數(shù)分別對應(yīng)于:

dom: 虛擬dom對應(yīng)的真實(shí)dom

attrs: 期望的最終鍵值屬性對

old: 當(dāng)前或者之前的屬性(從之前的VNode或者元素props屬性緩存中)

函數(shù)diffAttributes并不復(fù)雜,首先遍歷old中的屬性,如果當(dāng)前的屬性attrs中不存在是,則通過函數(shù)setAccessor將其刪除。然后將attr中的屬性賦值通過setAccessor賦值給當(dāng)前的dom元素。是否需要賦值需要同時滿足下滿三個條件:

屬性不能是children,原因children表示的是子元素,其實(shí)Preact在h函數(shù)已經(jīng)做了處理(詳情見系列文章第一篇),這里其實(shí)是不會存在children屬性的。

屬性也不能是innerHTML。其實(shí)這一點(diǎn)Preact與React是在這點(diǎn)是相同的,不能通過innerHTML給dom添加內(nèi)容,只能通過dangerouslySetInnerHTML進(jìn)行設(shè)置。

屬性在該dom中不存在 或者 如果當(dāng)該屬性不是value或者checked時,緩存的屬性(old)必須和現(xiàn)在的屬性(attrs)不一樣,如果該屬性是value或者checked時,則dom的屬性必須和現(xiàn)在不一樣,這么判斷的主要目的就是如果屬性值是value或者checked表明該dom屬于表單元素,防止該表單元素是不受控的,緩存的屬性存在可能不等于當(dāng)前dom中的屬性。那為什么不都用dom中的屬性呢?肯定是由于JavaScript對象中取屬性要比dom中拿到屬性的速度快很多。

  到這里我們有個地方需要注意的是,調(diào)用函數(shù)setAccessor時的第三個實(shí)參為old[name] = undefined或者old[name] = attrs[name],我們在前面說過,如果虛擬dom中的attributes發(fā)生改變時也需要將真實(shí)dom中的__preactattr_進(jìn)行更新,其實(shí)更新的過程就發(fā)生在這里,old的實(shí)參就是props = out[ATTR_KEY],所以更新old時也對應(yīng)修改了dom的緩存。

  我們最后需要關(guān)注的是函數(shù)setAccessor,這個函數(shù)比較長但是結(jié)構(gòu)是及其的簡單:
  

function setAccessor(node, name, old, value, isSvg) {
    if (name === "className") name = "class";

    if (name === "key") {
        // key屬性忽略
    }
    else if (name === "ref") {
        // 如果是ref 函數(shù)被改變了,以null去執(zhí)行之前的ref函數(shù),并以node節(jié)點(diǎn)去執(zhí)行新的ref函數(shù)
        if (old) old(null);
        if (value) value(node);
    }
    else if (name === "class" && !isSvg) {
        // 直接賦值
        node.className = value || "";
    }
    else if (name === "style") {
        if (!value || typeof value === "string" || typeof old === "string") {
            node.style.cssText = value || "";
        }
        if (value && typeof value === "object") {
            if (typeof old !== "string") {
                // 從dom的style中剔除已經(jīng)被刪除的屬性
                for (let i in old) if (!(i in value)) node.style[i] = "";
            }
            for (let i in value) {
                node.style[i] = typeof value[i] === "number" && IS_NON_DIMENSIONAL.test(i) === false ? (value[i] + "px") : value[i];
            }
        }
    }
    else if (name === "dangerouslySetInnerHTML") {
        //dangerouslySetInnerHTML屬性設(shè)置
        if (value) node.innerHTML = value.__html || "";
    }
    else if (name[0] == "o" && name[1] == "n") {
        // 事件處理函數(shù) 屬性賦值
        // 如果事件的名稱是以Capture為結(jié)尾的,則去掉,并在捕獲階段節(jié)點(diǎn)監(jiān)聽事件
        let useCapture = name !== (name = name.replace(/Capture$/, ""));
        name = name.toLowerCase().substring(2);
        if (value) {
            if (!old) node.addEventListener(name, eventProxy, useCapture);
        }
        else {
            node.removeEventListener(name, eventProxy, useCapture);
        }
        (node._listeners || (node._listeners = {}))[name] = value;
    }
    else if (name !== "list" && name !== "type" && !isSvg && name in node) {
        setProperty(node, name, value == null ? "" : value);
        if (value == null || value === false) node.removeAttribute(name);
    }
    else {
        // SVG元素
        let ns = isSvg && (name !== (name = name.replace(/^xlink:?/, "")));
        if (value == null || value === false) {
            if (ns) node.removeAttributeNS("http://www.w3.org/1999/xlink", name.toLowerCase());
            else node.removeAttribute(name);
        }
        else if (typeof value !== "function") {
            if (ns) node.setAttributeNS("http://www.w3.org/1999/xlink", name.toLowerCase(), value);
            else node.setAttribute(name, value);
        }
    }
}

  整個函數(shù)都是if-else的結(jié)構(gòu),首先看看各個參數(shù):

node: 對應(yīng)的dom節(jié)點(diǎn)

name: 屬性名

old: 該屬性之前存儲的值

value: 該屬性當(dāng)前要修改的值

isSvg: 是否為SVG元素

  然后看一下函數(shù)的流程:

如果屬性名為className,則屬性名修改為class,這一點(diǎn)Preact與React是不相同的,React對css中的類僅支持屬性名className,但Preact既支持className的屬性名也支持class,并且Preact更推薦使用class.

如果屬性名為key時,不做任何處理。

如果屬性名為class并且不是svg元素,則直接將值賦值給dom元素。

如果屬性名為style時,第一種情況是將字符串類型的樣式賦值給dom.style.cssText。如果value是空或者是字符串這么賦值非常能夠理解,但是為什么之前的屬性值old是字串符為什么也需要通過dom.style.cssText,經(jīng)過我的實(shí)驗(yàn)發(fā)現(xiàn)作用應(yīng)該是覆蓋之前通過cssText賦值的樣式(所以這里的代碼并不是if-else),而是兩個if的結(jié)構(gòu)。下面的第二種情況是value是對象類型,所進(jìn)行的操作是剔除取消的屬性,添加新的或者更改的屬性。

如果屬性是dangerouslySetInnerHTML,則將value中的__html值賦值給innerHtml屬性。

如果屬性是以on開頭,說明要綁定的是事件,因?yàn)槲覀冎繮react不同于React,并沒有采用事件代理的機(jī)制,所有的事件都會被注冊到真實(shí)的dom中。而且另一點(diǎn)與React不相同的是,如果你的事件名后添加Capture,例如onClickCapture,那么該事件將在dom的捕獲階段響應(yīng),默認(rèn)會在冒泡事件響應(yīng)。如果value存在則是注冊事件,否則會將注冊的事件移除。我們發(fā)現(xiàn)在調(diào)用addEventListener并沒有直接將value作為其第二個參數(shù)傳入,而是傳入了eventProxy:

function eventProxy(e) {
    return this._listeners[e.type](e);
}

  我們看到因?yàn)橛姓Z句(node._listeners || (node._listeners = {}))[name] = value,所以某個對應(yīng)事件的處理函數(shù)是保存在node._listeners對象中,因此當(dāng)函數(shù)eventProxy調(diào)用時,就可以觸發(fā)對應(yīng)的事件處理程序,其實(shí)這也算是一種簡單的事件代理機(jī)制,如果該元素對應(yīng)的某個事件處理程序發(fā)生改變時,也就不需要刪除之前的處理事件并綁定新的處理,只需要改變node._listeners對象存儲的對應(yīng)事件處理函數(shù)即可。
  

接下來為除了typelist以外的自有屬性進(jìn)行賦值或者刪除。其中函數(shù)setProperty為:

function setProperty(node, name, value) {
    try {
        node[name] = value;
    } catch (e) {
    }
}

  這個函數(shù)嘗試給為DOM的自有屬性賦值,賦值的過程可能在于IE瀏覽器和FireFox中拋出異常。所以這里有一個try-catch的結(jié)構(gòu)。

最后是為svg元素以及普通元素的非自有屬性進(jìn)行賦值或者刪除。因?yàn)閷τ诜亲杂袑傩允菬o非直接通過dom對象進(jìn)行設(shè)置的,僅可以通過函數(shù)setAttribute進(jìn)行賦值。

  到此為止,我們已經(jīng)基本全部分析完了Preact中diff算法的過程,我們看到Preact相比于龐大的React,短短數(shù)百行語句就實(shí)現(xiàn)了diff的功能并能達(dá)到一個相當(dāng)不錯的性能。由于本人能力所限,不能達(dá)到面面俱到,但希望這篇文章能起到拋磚引玉的作用,如果不正確指出,歡迎指出和討論~

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

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

相關(guān)文章

  • Preact了解一個React框架怎么實(shí)現(xiàn)(三): 組件

    摘要:組件渲染首先我們來了解組件返回的虛擬是怎么渲染為真實(shí),來看一下的組件是如何構(gòu)造的可能我們會想當(dāng)然地認(rèn)為組件的構(gòu)造函數(shù)定義將會及其復(fù)雜,事實(shí)上恰恰相反,的組件定義代碼極少。 前言   首先歡迎大家關(guān)注我的掘金賬號和Github博客,也算是對我的一點(diǎn)鼓勵,畢竟寫東西沒法獲得變現(xiàn),能堅(jiān)持下去也是靠的是自己的熱情和大家的鼓勵。  之前分享過幾篇關(guān)于React的文章: React技術(shù)內(nèi)幕: k...

    AlphaWatch 評論0 收藏0
  • 幫你讀懂preact源碼(一)

    摘要:是一個最小的庫,但由于其對尺寸的追求,它的很多代碼可讀性比較差,市面上也很少有全面且詳細(xì)介紹的文章,本篇文章希望能幫助你學(xué)習(xí)的源碼。建議與源碼一起閱讀本文。 作為一名前端,我們需要深入學(xué)習(xí)react的運(yùn)行機(jī)制,但是react源碼量已經(jīng)相當(dāng)龐大,從學(xué)習(xí)的角度,性價比不高,所以學(xué)習(xí)一個react mini庫是一個深入學(xué)習(xí)react的一個不錯的方法。 preact是一個最小的react mi...

    XboxYan 評論0 收藏0
  • 去哪兒網(wǎng)迷你React研發(fā)心得

    摘要:市面上竟然擁有多個虛擬庫。虛擬庫,就是出來后的一種新式庫,以虛擬與算法為核心,屏蔽操作,操作數(shù)據(jù)即操作視圖。及其他虛擬庫已經(jīng)將虛擬的生成交由與處理了,因此不同點(diǎn)是,虛擬的結(jié)構(gòu)與算法。因此虛擬庫是分為兩大派系算法派與擬態(tài)派。 去哪兒網(wǎng)迷你React是年初立項(xiàng)的新作品,在這前,去哪兒網(wǎng)已經(jīng)深耕多年,擁有QRN(react-native的公司制定版),HY(基于React的hybird方案)...

    pekonchan 評論0 收藏0
  • 幫你讀懂preact源碼(

    摘要:最后刪除新的樹中不存在的節(jié)點(diǎn)。而中會記錄對其做了相應(yīng)的優(yōu)化,節(jié)點(diǎn)的的情況下,不做移動操作。這種情況,在中得到了優(yōu)化,通過四個指針,在每次循環(huán)中先處理特殊情況,并通過縮小指針范圍,獲得性能上的提升。 上篇文章已經(jīng)介紹過idff的處理邏輯主要分為三塊,處理textNode,element及component,但具體怎么處理component還沒有詳細(xì)介紹,接下來講一下preact是如何處理...

    Warren 評論0 收藏0
  • FDCon2019 第4屆中國前端開發(fā)者千人峰會 - 《Omi - Cross-Frameworks

    摘要:用過的同學(xué)都知道,性能優(yōu)化的關(guān)鍵就是,最被詬病的也是這個,很多開發(fā)者也吐槽這個鉤子函數(shù),也可以配合不可變數(shù)據(jù)類型,直接進(jìn)行引用地址比較,來決定組件是否需要更新。 大家好,這次給大家講下 Omi 框架 以及即將發(fā)布的 Omim 大家有沒有數(shù)左邊的圖片里有多少個 Omi?Omi 團(tuán)隊(duì)很在意這里,特意數(shù)了下,有三個。Omi 團(tuán)隊(duì)希望 Omi 以后在各大會議里能夠印刷得更加大一些。今天給大家?guī)淼闹?..

    nifhlheimr 評論0 收藏0

發(fā)表評論

0條評論

張巨偉

|高級講師

TA的文章

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