《高性能JavaScript》学习笔记(2)——日更中

我说日更就日更,接着....今天从缓冲布局信息开始啦!

-------------------2016-7-22 21:09:12---------------------------

14、减少对布局信息的查询次数,查询时将他赋值给局部变量参与计算。

例子,在元素网右下方不断平移时,在timeout中可以写:

1     var current = myElement.offsetLeft;
2     current++;
3     myElement.style.left = current + ‘px‘;
4     myElement.style.top =  current + ‘px‘;
5     if(current > 500){
6         //stop animation
7     }

//拒绝下面的写法,每次移动都会查询一次偏移量,导致浏览器刷新渲染队列,非常耗时

1     myElement.style.left = myElement.offsetLeft + ‘px‘;
2     myElement.style.top =  myElement.offsetLeft + ‘px‘;
3     if(myElement.offsetLeft > 500){
4         //stop animation
5     }

15、大量的元素使用:hover之后,页面性能将降低,特别是IE8中。因此强烈建议,在数据量很大的表格中,减少鼠标在表上移动效果,减少高亮行的显示,使用高亮是个慢速过程CPU使用率会提高到80%-90%,尽量避免使用这种效果。

16、事件托管

讲到事件托管,首先我们来看一看冒泡机制:

DOM2级事件规定事件包括三个阶段: ① 事件捕获阶段 ② 处于目标阶段 ③ 事件冒泡阶段

图片引用来源:http://www.w3.org/TR/DOM-Level-3-Events/#event-flow

如下图的实验结果可以知道,当我们点击了inner之后,捕获和冒泡结果如上图的规律相同;

因此,因为每一个元素有一个或多个事件句柄与之相连时,可能会影响性能,毕竟连接每一个句柄都是有代价的,所以我们采用事件托管技术,在一个包装元素上挂接一个句柄,用于处理子元素发生的所有事件。

下面我们以如下的dom结构为例:

假如有一个ul,下面有很多个li:

1     <div>
2         <ul id="ulList">
3             <li id="item1"></li>
4             <li id="item2"></li>
5             <li id="item3"></li>
6             <li id="item4"></li>
7             <li id="item5"></li>
8         </ul>
9     </div>

当某个Li被点击的时候需要触发相应的处理事件。我们通常的写法,是为每个Li都添加onClick的事件监听。

 1 function addListenersLi(liNode){
 2     liNode.onclick = function clickHandler(){}
 3 }
 4
 5 window.onload = function(){
 6     var ulNode = document.getElementById("ulList");
 7     var liNodes = ulNode.getElementByTagName("li");
 8     for(var i=0, l = liNodes.length; i < l; i++){
 9         addListeners4Li(liNodes[i]);
10     }
11 }

如果li足够多,或者对于li的操作特别频繁,为每一个li绑定一个点击事件将会特别影响性能,因为在此期间,你需要访问和修改更多的DOM节点,事件的绑定过程发生在onload事件中,绑定本身也非常耗时;同时,浏览器需要保存每个句柄的记录,很占用内存。重点是有些绑定了还不一定会用着,并不是100%的按钮或链接都会被点到的哟!

因此,采用事件托管更为高效,当事件被抛到更上层的父节点的时候,我们通过检查事件的目标对象(target)来判断并获取事件源Li。下面的代码可以完成我们想要的效果:

 1         var oul = document.getElementById(‘ulList‘);
 2         oul.addEventListener(‘click‘,function(e){
 3             var e = e || window.event;
 4             var target = e.target || e.srcElement;
 5
 6             console.log(target.nodeName);
 7             if(target.nodeName == ‘LI‘){
 8                 //事件真正的处理程序
 9                 alert(target.id);
10                 e.preventDefault();
11                 e.stopPropagation();
12             }else{
13                 alert(456);
14             }
15
16         });

太晚了,睡觉了,明天再继续,有问题,请多多指教!

时间: 2024-10-17 09:12:12

《高性能JavaScript》学习笔记(2)——日更中的相关文章

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

这篇笔记的内容主要涉及js的脚本位置,如何加载js脚本和脚本文件执行的问题,按照自己的理解结合高性能JavaScript整理出来的 javascript是解释性代码,解释性代码需要经历转化成计算机指令的过程,这个过程就会带来一定的性能损耗,所以在js中做性能的优化是必须的 javascript的阻塞特性:浏览器在执行js代码的时候,不能做其他的任何事情,因为浏览器使用单一的进程来处理用户界面的刷新和javascript的脚本执行,也就是说什么时候执行js脚本影响着用户对页面的使用体验(之所以js

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

参考 高性能javascript javascript高级程序设计 ajax基础  ajax技术的核心是XMLHttpRequest对象(XHR),通过XHR我们就可以实现无需刷新页面就能从服务器端读取数据 var xhr = new XMLHttpRequest(); //只支持IE7以及更高的版本 xhr.onreadystatechange = function() { if(xhr.readyState == 4) { if( (xhr.status >= 200 && xh

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

参考高性能javascript for in 循环  使用它可以遍历对象的属性名,但是每次的操作都会搜索实例或者原型的属性 导致使用for in 进行遍历会产生更多的开销 书中提到不要使用for in 遍历数组 1 首先for in 会查找原型链上的属性 var arr = [1,2,3]; Array.prototype.a = "test"; for(var i in arr) { console.log(i); console.log(typeof i); }//在这里例子中会发

高性能javascript学习笔记系列(5) -快速响应的用户界面

参考高性能javascript 理解浏览器UI线程  用于执行javascript和更新用户界面的进程通常被称为浏览器UI线程  UI线程的工作机制可以理解为一个简单的队列系统,队列中的任务按顺序执行 <button onclick="handleClick()">click</button> <script type="text/javascript"> function handleClick() { var div = do

高性能javascript 学习笔记(1)

加载和运行 管理浏览器中的javascript代码是个棘手的问题,因为代码运行阻塞了其他浏览器处理过程,诸如用户绘制,每次遇到<script>标签,页面必须停下来等待代码下载(如果是外部的)并执行,然后再继续处理页面其他部分.但是,有几种方法可以减少javascript对性能的影响: 将所有<script>标签放置在页面的底部,紧靠body关闭标签</body>的上方,此法可以保证在脚本运行之前完成解析. 将脚本成组打包.页面的<script>标签越少,页面

javascript学习笔记-2:jQuery中$(&quot;xx&quot;)返回值探究

最近在写一个jQuery插件的时候,需要用到一个条件: 一组img标签,每一个元素都需要被它前面的元素值src替换,如果是第一个(序列为0)则其值为最后一个元素值,如果是最后一个,那么其值为第一个元素值,以此形成一个闭环. 为此,我使用了三元运算符?:,其表达式为:var next=$(this).next()?$(this).next():imageItems.first(); 运行测试发现如下问题,当运行到数组最后一个元素时,其next是始终不会是这一组img标签的第一个,为此对$(this

javascript学习笔记(三) string对象中的正则表达式

1. search返回匹配到的位置(-1找不到) var str = 'html js' var pattern = /js/ str.search(pattern)    --------->5 2. Match str.match(pattern) --------['js'] /js/g -----> ["js", "js",,"js"] match vs exec match:非全局的情况下才会返回分组中匹配到的内容,全局匹配

JavaScript学习笔记——jquery中html()、text()、val()的区别

.html()用为读取和修改元素的HTML标签 .text()用来读取或修改元素的纯文本内容 .val()用来读取或修改表单元素的value值. 这三个方法功能上的对比 .html(),.text(),.val()三种方法都是用来读取选定元素的内容:只不过.html()是用来读取元素的HTML内容(包括其Html标签),.text()用来读取元素的纯文本内容,包括其后代元素,.val()是用来读取表单元素的"value"值.其中.和.text()方法不能使用在表单元素上,而.val()

JavaScript学习笔记【2】表达式和运算符、语句、对象

笔记来自<JavaScript权威指南(第六版)> 包含的内容: 表达式和运算符 语句 对象 表达式和运算符 数组直接量中的列表逗号之间的元素可以省略,这时省略的空位会填充值undefined.元素列表末尾可以留下单个逗号,这时并不会创建一个新的值为undefined元素. 属性访问表达式,.identifier的写法只适用于要访问的属性名称是合法的标识符,并且需要知道要访问的属性的名字.如果属性名称是一个保留字或者包含空格和标识符,或是一个数字(对于数组来说),则必须使用方括号的写法.当属性