zoukankan      html  css  js  c++  java
  • 小tip: DOM appendHTML实现及insertAdjacentHTML

    一、无人不识君

    据说今天是邓丽君奶奶会见马克思的日子,所谓“无人不识君”就多了份“无人不识邓丽君”之意。

    JS中有很多基本DOM方法,例如createElementparentNode等,其中,appendChild方法是相当地常用与熟知,可谓是DOM节点方法中的“无人不识君”!

    appendChild的作用是在指定元素节点的最后一个子节点之后添加节点。好记又好用,大家都喜欢。

    appendChild方法就像是直接买饺子,实际上,我们还经常会遇到买饺子皮和馅自己包的情况。放在HTML中解释就是不是append节点,而是append构成节点的HTML字符代码。

    例如,我们点击“更多评论”按钮,需要ajax加载评论相关HTML代码并插入到页面中,此时,appendChild显然就不如appendHTML方法来得方便。

    二、如何实现appendHTML方法?

    如果是纯粹没有脚本等行为的容器。 我们直接innerHTML拼接就可以了。例如,append一个图片HTML片段,我们可以:

    container.innerHTML = container.innerHTML + '<img src="zhanghanyun.jpg" />';

    好东西喜欢留到最后吃,重点都是留到最后讲。因此,上面的,大家都懂的,就是走过场的,只在特殊情况用用(单纯HTML处理),如果真要构造一个通用的appendHTML方法,innerHTML拼接显然是没有市场的。 //zxx: 如果是面试,一定要先说重点,不要铺垫酝酿什么的……

    考虑到容器的原HTML极可能包含事件,因此,实现append效果的时候一定不能干扰之前的内容,那我们该如何实现呢?

    不卖关子了,还是要借助appendChild方法。我们把HTML字符串转换成节点,然后通过appendChild方法载入进去。

    如何将HTML字符串(假设字符串变量名为html)转换成节点呢?如下操作即可:

    // 创建div节点
    var div = document.createElement("div");
    // 装载html字符串
    div.innerHTML = html;
    // 此时div.childNodes就是我们需要的节点了!
    return div.childNodes;
    

    下面要做的就是将这些节点append进去,下面是我们自然理解的实现,遍历:

    var nodes = div.childNodes;
    for (var i=0, length=nodes.length; i<length; i+=1) {
       // 容器container加载克隆的节点 - 克隆的作用是保证nodes的完整
       container.appendChild(nodes[i].cloneNode(true)); 
    }
    

    上面代码功能虽然实现,但是性能no, no, no. 尤其低版本IE下,缺少优化的机制,每次appendChild造成的回流与渲染会让浏览器high到叫的。

    对于DOM节点插入,大家应该都熟知“文档片段优化法”。具体来讲,就是使用document.createDocumentFragment()创建一个文档片段,然后,把节点一个一个append到这个片段中,回到页面上的时候,直接append这个文档片段就可以了-只有一次。

    形象点讲就是:原本拿个小刀一点一点割小弟弟改成直接一把快刀咔嚓成东方姑娘。一次性解决痛苦!

    岳不群自宫图 张鑫旭-鑫空间-鑫生活

    代码解释就是:

    var nodes = div.childNodes
        // 我就是那把快刀,
        , fragment = document.createDocumentFragment();
    
    for (var i=0, length=nodes.length; i<length; i+=1) {
       // 文档片段加载克隆的节点
       fragment.appendChild(nodes[i].cloneNode(true)); 
    }
    // 一刀来个痛快
    container.appendChild(fragment);
    

    于是,我们在HTML元素原型上扩展,可以让高端点的浏览器(IE9+, …)都有了appendHTML方法。

    HTMLElement.prototype.appendHTML = function(html) {
        var divTemp = document.createElement("div"), nodes = null
            // 文档片段,一次性append,提高性能
            , fragment = document.createDocumentFragment();
        divTemp.innerHTML = html;
        nodes = divTemp.childNodes;
        for (var i=0, length=nodes.length; i<length; i+=1) {
           fragment.appendChild(nodes[i].cloneNode(true));
        }
        this.appendChild(fragment);
        // 据说下面这样子世界会更清净
        nodes = null;
        fragment = null;
    };

    demo实例
    为兼容IE6~8浏览器,直接元素原型扩展的方法并不实用,我们可能还是要中规中矩的使用,例如:

    var appendHTML  = function(el, html) {
        // 全部都是一样的,除了下面这个 this → el
        el.appendChild(fragment);
    };

    您可以狠狠地点击这里:ajax加载HTML并应用appendHTML载入页面demo

    点击查看个更多评论
    更多评论相关的HTML代码 评论载入后的源代码查看 张鑫旭-鑫空间-鑫生活

    三、有木有prependHTML方法?

    appendHTML是在容器的最后加载HTML,那可不可以实现在容器前面加载HTML的方法呢,姑且命名为prependHTML

    自然可以,差别在于不是借助appendChild, 而是insertBefore方法。只要做如下修改就可以了:

    this.appendChild(fragment);
    // 变成↙
    this.insertBefore(fragment, el.firstChild);

    完整代码如下(单纯的兼容方法):

    var prependHTML = function(el, html) {
        var divTemp = document.createElement("div"), nodes = null
            , fragment = document.createDocumentFragment();
    
        divTemp.innerHTML = html;
        nodes = divTemp.childNodes;
        for (var i=0, length=nodes.length; i<length; i+=1) {
           fragment.appendChild(nodes[i].cloneNode(true));
        }
        // 插入到容器的前面 - 差异所在
        el.insertBefore(fragment, el.firstChild);
        // 内存回收?
        nodes = null;
        fragment = null;
    };

    您可以狠狠地点击这里:prependHTML方法加载更老的评论demo

    显示更早的评论操作演示

    为节省代码,前加载HTML可以不使用prependHTML, 而是还是命名为appendHTML方法,不过通过另外一个参数控制加载的位置。处理如下:

    var appendHTML  = function(el, html, where) {
        where = where || "bottom";
        // where参数bottom(默认)或者before
        where !== "before"? el.appendChild(fragment) : el.insertBefore(fragment, el.firstChild);;
    };

    MooTools框架中的DOM方法很多都是这个思路。

    四、结束语

    文章这东西啊,写长了有时候反而不好,为什么呢?大家都很忙的,啰哩吧嗦的文章,大家都会不由自主加快鼠标滚动,贡献的价值就会打折扣。

    其实本文只想点到为止,貌似还是啰嗦了点…………好像还行,总之以后会多加注意的。

    本文提供appendHTML方法自己拙计写的,可能不是最优的实现,欢迎能人指点。如果您JS刚入门,研究下appendHTML实现对于熟悉DOM相关处理还是很有帮助的。

    备注:
    ① 大部分浏览器对元素几何改变时候的重排做了优化。据说是这样子,一定时间内本应多次重排的改变,浏览器会hold住,仅一次重排。其中如果使用分离的一步处理过程,例如计时器,依然多次重排。例如,当我们应用transition动画的时候,希望从0px变化到100px. 你如果如下代码:

    dom.style.left = "0px";
    dom.style.left = "100px";

    元素是不会从0~100像素动画的,因为现代浏览器有自己的优化机制,它只会处理后面的dom.style.left = "100px",使用定时器可以阻断这种优化,实现我们想要的过渡动画效果,如下:

    dom.style.left = "0px";
    setTimeout(function() { dom.style.left = "100px"; }, 20);

    五、强烈补充

    @hafeyang 在评论中提到了insertAdjacentHTML方法,这个方法我很早时候曾见到过,据说有兼容性问题(FireFox浏览器),因此,没有怎么搭理。

    世事境迁,今天我重新测试,在控制台输入document.documentElement.insertAdjacentHTML发现不是undefined了,看来我out了。

    忙去MDN查看一番,发现FireFox浏览器在FireFox 8版本开始支持insertAdjacentHTML方法了。因此,这使得目前使用insertAdjacentHTML实现HTML片段插入效果成为了可能。

    语法如下:

    element.insertAdjacentHTML(position, html);

    position是相对于element元素的位置,并且只能是以下的字符串之一:

    beforebegin
    在 element 元素的前面。
    afterbegin
    在 element 元素的第一个子元素前面。
    beforeend
    在 element 元素的最后一个子元素后面。
    afterend
    在 element 元素的后面。

    html是字符串被解析成HTML或XML插入到DOM树中。

    其中beforeend参数就是我们需要的appendHTML效果。

    您可以狠狠地点击这里:insertAdjacentHTML方法beforeend参数demo

    IE6浏览器下insertAdjacentHTML方法beforeend参数IE6下效果截图

    我们如何牢牢记住insertAdjacentHTML这个词呢?

    从中文语义来看,insert(插入)Adjacent(邻近)HTML。就我自己来说,insert是已经熟记的,Adjacent不太熟,会这么记忆,“广告夹生的”——Ad为广告缩写,jacent读音近似中文“夹生的”。so…

  • 相关阅读:
    2014年互联网发展趋势如何
    服务器出现阶段性错误
    用互联网思想武装自己
    杭州互联网公司汇总
    互联网牛人网址大全
    ffmpeg开发指南
    Windows下FFmpeg快速入门
    FFmpeg介绍及参数详细说明
    windows 下FFMPEG的编译方法 附2012-9-19发布的FFMPEG编译好的SDK下载
    FFMPEG视音频编解码零基础学习方法 【荐】
  • 原文地址:https://www.cnblogs.com/xiaoleiel/p/8308791.html
Copyright © 2011-2022 走看看