高性能JavaScript之DOM编程
我们知道。DOM是用于操作XML和HTML文档的应用程序接口,用脚本进行DOM操作的代价非常昂贵。
有个贴切的比喻。把DOM和JavaScript(这里指ECMScript)各自想象为一个岛屿,它们之间用收费桥梁连接,ECMAScript每次訪问DOM。都要途径这座桥,并交纳“过桥费”,訪问DOM的次数越多,费用也就越高。因此。推荐的做法是尽量降低过桥的次数,努力待在ECMAScript岛上。我们不可能不用DOM的接口,那么。如何才干提高程序的效率?
1、DOM訪问与改动
訪问DOM元素是有代价的(“过桥费”你懂的)。改动元素代价更是昂贵。由于它会导致浏览器又一次计算页面的几何变化(重排和重绘)。
当然最坏的情况是在循环中訪问或者改动元素,看以下两段代码:
var times = 200;
// code1
console.time(1);
for(var i = 0; i < times; i++) {
document.getElementById('myDiv1').innerHTML += 'a';
}
console.timeEnd(1);
// code2
console.time(2);
var str = '';
for(var i = 0; i < times; i++) {
str += 'a';
}
document.getElementById('myDiv2').innerHTML = str;
console.timeEnd(2);
结果第一次执行的时间竟然是第二次的几十倍!(chrome 版本号 44.0.2403.130 m)
1: 5.538ms
2: 0.111ms
第一段代码的问题在于,每次循环迭代,该元素都会被訪问两次:一次读取innerHTML的值,还有一次重写它。也就是说。每次循环都在过桥(重排和重绘将在下一篇解说)!结果充分表明,訪问DOM的次数越多,代码的执行速度越慢。因此。能降低DOM訪问的次数则尽量降低,尽量留在ECMAScript这端处理。
2、HTML集合 & 遍历DOM
操作DOM还有一个耗能点就是遍历DOM。一般我们会收集一个HTML集合,比方用getElementsByTagName(),或者用document.links等,我想大家对此都不陌生。收集的结果是一个相似数组的集合。它处于一种“实时状态”实时存在,这意味着当底层文档对象更新时。它也会自己主动更新。怎么讲?非常easy举个栗子:
<body>
<ul id='fruit'>
<li> apple </li>
<li> orange </li>
<li> banana </li>
<li> stabery </li>
</ul>
</body>
<script type="text/javascript">
var lis = document.getElementsByTagName('li');
var peach = document.createElement('li');
peach.innerHTML = 'peach';
document.getElementById('fruit').appendChild(peach);
console.log(lis.length); // 4
</script>
而这正是低效之源!
非常easy,跟数组的优化操作一样,缓存个length变量就ok了(读取一个集合的length比读取一个普通数组的lengh要慢非常多,由于每次都要查询):
console.time(0);
var lis0 = document.getElementsByTagName('li');
var str0 = '';
for(var i = 0; i < lis0.length; i++) {
str0 += lis0[i].innerHTML;
}
console.timeEnd(0);
console.time(1);
var lis1 = document.getElementsByTagName('li');
var str1 = '';
for(var i = 0, len = lis1.length; i < len; i++) {
str1 += lis1[i].innerHTML;
}
console.timeEnd(1);
我们看看性能提升能有多少?
0: 0.974ms
1: 0.664ms
当集合的长度大的时候(demo是1000)。性能提升还是非常明显的。
而《高性能JavaScript》提出了还有一个优化策略。它指出,“由于遍历数组比遍历集合快。因此假设先将集合元素复制到数组中,那么訪问它的属性会更快”,经过測试,并没有非常好地发现这个规律。所以还是不要多此一举了,測试代码例如以下:(有疑义欢迎与我交流探讨)
console.time(1);
var lis1 = document.getElementsByTagName('li');
var str1 = '';
for(var i = 0, len = lis1.length; i < len; i++) {
str1 += lis1[i].innerHTML;
}
console.timeEnd(1);
console.time(2);
var lis2 = document.getElementsByTagName('li');
var a = [];
for(var i = 0, len = lis2.length; i < len; i++)
a[i] = lis2[i];
var str2 = '';
for(var i = 0, len = a.length; i < len; i++) {
str2 += a[i].innerHTML;
}
console.timeEnd(2);
后来反思了一下,意思大概是这样:
console.time(2);
var lis2 = document.getElementsByTagName('li');
var a = [];
for(var i = 0, len = lis2.length; i < len; i++)
a[i] = lis2[i];
str2 += a[i].innerHTML;
console.timeEnd(2);
这样确实有非常显著的区别哦:
1: 2.615ms
2: 0.164ms
3、querySelector()和querySelectorAll()
本节的最后介绍两个原生DOM方法。querySelector()和querySelectorAll()。相信大家都不陌生,前者返回一个数组(注意。它们的返回值不像HTML集合一样会动态变化),后者返回匹配的第一个元素。好吧,事实上并非所有时候它的性能都优于前者的HTML集合遍历。
console.time(1);
var lis1 = document.getElementsByTagName('li');
console.timeEnd(1);
console.time(2);
var lis2 = document.querySelectorAll('li');
console.timeEnd(2);
// 1: 0.038ms
// 2: 3.957ms
可是由于它是相似CSS的选择方法。所以在做组合选择的时候,效率会提升。又方便。比方做例如以下的组合查询:
var elements = document.querySelectorAll('#menu a');
var elements = document.querySelectorAll('div.warning, div.notice');
以上就是关于高性能JavaScript DOM编程的所有内容。希望大家能够理解,对大家的学习有所帮助。
高性能JavaScript之DOM编程的更多相关文章
- 高性能Javascript(2) DOM编程
第三部分 DOM编程 文档对象模型(DOM)是一个独立于语言的,使用XML和HTML文档操作的应用程序接口(API).在浏览器中,主要与HTML文档打交道,在网页应用中检索XML文档也很常见.DOM ...
- 高性能JavaScript(DOM编程)
首先什么是DOM?为什么慢? DOM:文档对象模型,是一个独立于语言的,用于操作XML和HTML文档的程序接口(API) 用脚本进行DOM操作的代价很昂贵.那么,怎样才能提高程序的效率? 1.DOM访 ...
- HTML、css、javascript、DOM编程
HTML.css.javascript.DOM编程 一.Html 1.1html概述 Html就是超文本标记语言的简写,是最基础的网页语言,其代码都是由标签所组成,是通过标签来定义的语言,代码不需要区 ...
- JavaScript的DOM编程--12--innerHTML属性
innerHTML属性: 1). 浏览器几乎都支持该属性, 但不是 DOM 标准的组成部分. innerHTML 属性可以用来读, 写某给定元素里的 HTML 内容 <html> < ...
- JavaScript的DOM编程--01--js代码的写入位置
DOM:Document Object Model(文本对象模型) D:文档 – html 文档 或 xml 文档 O:对象 – document 对象的属性和方法 M:模型 DOM 是针对xml(h ...
- JavaScript的DOM编程--11--插入节点
插入节点: 1). insertBefore(): 把一个给定节点插入到一个给定元素节点的给定子节点的前面 var reference = element.insertBefore(newNode,t ...
- JavaScript的DOM编程--10--删除节点
1). removeChild(): 从一个给定元素里删除一个子节点 var reference = element.removeChild(node); 返回值是一个指向已被删除的子节点的引用指针. ...
- JavaScript的DOM编程--09--节点的替换
节点的替换: 1). replaceChild(): 把一个给定父元素里的一个子节点替换为另外一个子节点 var reference = element.replaceChild(newChild,o ...
- JavaScript的DOM编程--08--复习
<html> <head> <meta http-equiv="Content-Type" content="text/html; char ...
随机推荐
- python的开发环境配置-Eclipse-PyDev插件安装
安装PyDev插件的两种安装方法: 1.百度搜索PyDev 2.4.0.zip,下载后解压,得到Plugins和Feature文件夹,复制两文件夹到Eclipse目录,覆盖即可. 插件的版本要对应py ...
- LeetCode(7):颠倒整数
Easy! 题目描述:给定一个范围为 32 位 int 的整数,将其颠倒. 例1: 输入:132 输出:321 例2: 输入:-123 输出:-321 例3: 输入:120 输出:21 注意:假设我们 ...
- Laravel Cache 的缓存文件在到期后是否会自动删除
验证缓存文件是否会自动删除的目的是,防止产生大量的缓存文件,占满磁盘.因为,我最近越来越多的使用 cache 来缓存各类 token. 使用的是 file 作为 CACHE_DRIVER CACHE_ ...
- laravel 集合
最近一直在用laravel框架,比较喜欢laravel的ORM(通常我们理解的Model)...但是默认情况下,Eloquent 查询的结果总是返回 Collection 实例...所有不得不了解co ...
- 目标检测-yolo
论文下载:http://arxiv.org/abs/1506.02640 代码下载:https://github.com/pjreddie/darknet 1.创新点 端到端训练及推断 + 改革区域建 ...
- poj1177 矩形周长并
线段树扫描线的模板题,一个月前写的发现忘了一些还是要看看以前的博客呀! /* 思路:数据小不用离散化处理,线段树叶子结点维护一个区间 */ #include<iostream> #incl ...
- Windows安装使用Openssl
1.什么是openssl? 2.下载安装 三方下载地址 备用64位和32位下载地址 选择32位或者64位合适的版本下载,例如Win64OpenSSL_Light-1_0_2h.exe: 设置环境变量, ...
- python 全栈开发,Day73(django多表添加,基于对象的跨表查询)
昨日内容回顾 多表方案: 如何确定表关系呢? 表关系是在2张表之间建立的,没有超过2个表的情况. 那么相互之间有2条关系线,先来判断一对多的关系. 如果其中一张表的记录能够对应另外一张表的多条记录,那 ...
- ahoi2009维护序列
链接:https://www.luogu.org/problemnew/show/P2023 裸的线段树维护+* 代码: #include <bits/stdc++.h> using na ...
- C#编程语法积累(二)
9.Lambda表达式 [1]Lambda表达式缩写推演,如下图: [2]Lambda语句:=>右边有一个语句块(大括号"{}"):Lambda表达式:=>右边只有一个 ...