参考 高性能javascript

文档对象模型(DOM)是独立于语言的,用于操作XML和HTML文档的程序接口API,在浏览器中主要通过DOM提供的API与HTML进行交互,浏览器通常会把DOM和javascript独立实现,这意味着当我们通过javascript操作DOM的时候势必的产生性能上的开销,因此提升页面性能的的思路是尽可能的减少DOM操作或者优化DOM操作

DOM的访问与修改

(1)减少频繁的读取DOM元素

(1)for(var i = 0;i < 10;i++) {
document.getElementById("test") += "a";
}
//上面的代码每次循环都进行了读取
(2)var content = "";
for(var i = 0;i < 10;i++) {
content += "a";
}
document.getElementById("test").innerHTML = content; //只有在最后一次通过innerHTML进行了读取 当我们更新一段html是使用innerHTML代替原始的dom操作

(2)节点克隆

当需要创建大量相似的节点的时候,通过element.clonenode()复制相应的节点,来代替重复的创建相应的节点来提升一部分的性能

(3)处理HTML集合

当我们通过document.getElementsByTagName()   document.getElementsByClassName() document.getElementsByName()  返回的是nodelist  它是一种类数组的集合,我们能通过下标的方式查找到集合中的元素

同样的通过下面的属性得到的也是html集合

document.images document.links document.forms document.forms[0].elements

操作html集合会出现性能的问题是html集合是维持实时的状态,也就是当你在文档中添加一个元素的时候,这个时候你在通过特定的方式获取html集合的长度的时候它的长度是实时变化的,html集合与文档处于连接的状态,获取html的信息的时候需要进行查询操作。

处理html集合元素有大致两种方式:

3.1将html集合转换成数组,因为遍历数组相对于遍历集合更快

  function toArray(coll) {
for(var i = 0,a = [],length = coll.length;i < len;i++) {
a[i] = coll[i];
}
return a;
}

3.2减少集合的查询操作,缓存局部变量

function test(coll) {
var length = coll.length;
var id = "";
for(var i = 0;i <length;i+=1) {
var el = coll[i];
id = el.getAttribute("id");
console.log(id);
}
}

(4)选取dom元素

4.1 使用能区分元素节点的选择API

属性名 被替代的属性
children childNodes
childElementCount childNodes.length
firstElementChild firstChild
lastElementChild  lastChild
nextElementSibling nextSibling
previousElementSibling previousSibling

上面的第一列属性只会返回相应的元素节点(两个节点之间的空格也会被识别成文本节点)  当我们通过第二列属性进行操作的时候,就需要在进行一步过滤的操作,增加了开销

4.2 使用选择器API

当需要进行大量的组合操作的时候,通过document.querySelectorAll()来进行元素的选择,通过这种方式获取的是一个nodelist,但它不与文档进行实时的更新,是一种静态的列表

(5)重绘与重排

当下载完页面的组件后,就会生成两个页面结构 DOM树  表示页面的结构  渲染树 表示DOM节点如何被显示   每一个在DOM树中需要被显示的元素在渲染树中至少存在一个对应的节点(在渲染树中的节点称为帧或者盒),当DOM的变化影响了元素的几何属性(宽高)的时候,这样浏览器就会重新计算元素的几何尺寸,其他元素的几何尺寸也会受到相应的影响,浏览器会使渲染树中受到影响的部分失效并重新构造渲染树,这个过程叫做重排,完成重排后浏览器重新绘制受到影响的部分,这个过程就叫做重绘

重绘和重排都是代价昂贵的操作,所以优化的方案是最小化重绘和重排

重排发生的时机

  • 添加或删除可见的DOM元素
  • 元素位置的改变
  • 元素尺寸的变化
  • 内容的改变(例如文本的改变或者图片内容的改变影响了元素的几何属性)
  • 页面渲染器的初始化
  • 浏览器窗口尺寸的改变

浏览器会优化重排的过程,通过队列的方式修改并批量的执行重拍的过程,但是当我们使用一些属性或者方法的时候,会强制的刷新渲染队列触发重排,当我们频繁的进行这些操作的时候,就会造成性能的开销

优化的思路:

(1)集中操作  将设置样式和获取样式进行分离(不要在布局信息改变的时候去进行查询操作),并且尽可能的合并操作

var el = document.getElementById("test");
el.style.cssText = "border-left:1px;border-right:2px;border-bottom:3px;";

(2)批量的修改DOM的时候,可以通过下面的方式减少重绘与重排

  1. 使元素脱离文档流
  2. 对齐应用修改
  3. 将元素带回文档流

脱离文档流的方案

  1. 隐藏元素 应用修改 显示元素
  2. 使用文档碎片(使用document.createDocumentFragment()创建的文档碎片添加到文档的时候,就将文档碎片的子节点进行添加而不是文档碎片本身)
  3. 将原始元素拷贝进行修改,然后替换元素元素

(3)缓存布局信息 这个思路有点像处理集合元素的时候的方案,通过缓存局部的布局信息,能减少强制刷新渲染队列的次数,也就减少了重排

(4)动画元素的处理 其实思路是一样的 动画有可能会造成大量其他元素位置改变,当动画开始的时候,让它脱离文档流,进行动画的展示,动画完结后让它回到文档流

高性能javascript学习笔记系列(3) -DOM编程的更多相关文章

  1. 高性能javascript学习笔记系列(5) -快速响应的用户界面和编程实践

    参考高性能javascript 理解浏览器UI线程  用于执行javascript和更新用户界面的进程通常被称为浏览器UI线程  UI线程的工作机制可以理解为一个简单的队列系统,队列中的任务按顺序执行 ...

  2. 高性能javascript学习笔记系列(1) -js的加载和执行

    这篇笔记的内容主要涉及js的脚本位置,如何加载js脚本和脚本文件执行的问题,按照自己的理解结合高性能JavaScript整理出来的 javascript是解释性代码,解释性代码需要经历转化成计算机指令 ...

  3. 高性能javascript学习笔记系列(2)-数据存取

    参考 高性能javascript Tom大叔深入理解javascript系列 相关概念 1.执行上下文   当控制器转到ecmascript可执行代码的时候,就会进入一个执行上下文,执行上下文是以堆栈 ...

  4. 高性能javascript学习笔记系列(6) -ajax

    参考 高性能javascript javascript高级程序设计 ajax基础  ajax技术的核心是XMLHttpRequest对象(XHR),通过XHR我们就可以实现无需刷新页面就能从服务器端读 ...

  5. 高性能javascript学习笔记系列(4) -算法和流程控制

    参考高性能javascript for in 循环  使用它可以遍历对象的属性名,但是每次的操作都会搜索实例或者原型的属性 导致使用for in 进行遍历会产生更多的开销 书中提到不要使用for in ...

  6. JavaScript学习笔记系列2:Dom操作(一)

    一.什么是Dom? DOM------>Document Object Model 直接翻译就是文档对象模型. DOM------>定义了表示和修改文档所需的对象.这些对象的行为和属性以及 ...

  7. 《高性能javascript》 领悟随笔之-------DOM编程篇(二)

    <高性能javascript> 领悟随笔之-------DOM编程篇二 序:在javaSctipt中,ECMASCRIPT规定了它的语法,BOM实现了页面与浏览器的交互,而DOM则承载着整 ...

  8. 《高性能javascript》 领悟随笔之-------DOM编程篇

    <高性能javascript> 领悟随笔之-------DOM编程篇一 序:在javaSctipt中,ECMASCRIPT规定了它的语法,BOM实现了页面与浏览器的交互,而DOM则承载着整 ...

  9. Javascript学习笔记三——操作DOM(二)

    Javascript学习笔记 在我的上一个博客讲了对于DOM的基本操作内容,这篇继续巩固一下对于DOM的更新,插入和删除的操作. 对于HTML解析的DOM树来说,我们肯定会时不时对其进行一些更改,在原 ...

随机推荐

  1. 实现CSS等分布局的4种方式

    × 目录 [1]float [2]inline-block [3]table[4]flex 前面的话 等分布局是指子元素平均分配父元素宽度的布局方式,本文将介绍实现等分布局的4种方式 思路一: flo ...

  2. App乱世,3721离我们有多远

    [总结]根据众多网友的评论,看来还是WP比较给力,IOS太贵...会对手机进行优化,安卓还行,如果给中老年人用WP比较好 声明:合理讨论,禁止骂人言论,本人也不是5毛党,发表下个人看法而已. 快过年了 ...

  3. 使用ZeroClipboard解决跨浏览器复制到剪贴板的问题

    Zero Clipboard的实现原理Zero Clipboard 利用透明的Flash让其漂浮在复制按钮之上,这样其实点击的不是按钮而是 Flash ,这样将需要的内容传入Flash,再通过Flas ...

  4. 用Log Parser Studio分析IIS日志

    发现一个强大的图形化IIS日志分析工具——Log Parser Studio,下面分享一个实际操作案例. 1. 安装Log Parser Studio a) 需要先安装Log Parser,下载地址: ...

  5. vs切换当前编辑文件时自动定位目录树

    在编辑区,切换当前编辑文件时(单击.cpp或.h文件选项卡),"解决方案资源管理器"目录树会自动定位当前编辑的文件,并以灰色标识,当一个解决方案中的工程数目数目很多,每个工程下面又 ...

  6. dom4j的读写xml文件,读写xml字符串

    百度了一些博客,大同小异,在选取jar包工具的时候大概看了下,大抵是jdom原始,dom4j优秀.于是做了些练习. 参考:http://www.cnblogs.com/mengdd/archive/2 ...

  7. Java实现过滤中文乱码

    最近在日志数据清洗时遇到中文乱码,如果只要有非中文字符就将该字符串过滤掉,这种方法虽简单但并不可取,因为比如像Xperia™主題.天天四川麻将Ⅱ这样的字符串也会被过滤掉. 1. Unicode编码 U ...

  8. Django--models一对多

    一对多--foreignkey 应用场景 当一张表中创建一行数据时,有一个单选的下拉框(可以被重复选择) 例如:创建用户信息时候,需要选择一个用户类型[普通用户][金牌用户][铂金用户]等. 创建表 ...

  9. postgres中的视图和物化视图

    视图和物化视图区别 postgres中的视图和mysql中的视图是一样的,在查询的时候进行扫描子表的操作,而物化视图则是实实在在地将数据存成一张表.说说版本,物化视图是在9.3 之后才有的逻辑. 比较 ...

  10. Android Studio快捷键每日一练(5)

    原文地址:http://www.developerphil.com/android-studio-tips-of-the-day-roundup-5/ 42.Enter键和Tab键补全 快捷键:Ent ...