做好七件事帮你提升jQuery的性能


Posted in Javascript onFebruary 06, 2014

1. Append Outside of Loops

凡是触及到DOM都是有代价的。如果你向DOM当中附加大量的元素,你会想一次性将它们全部附加进来,而不是分多次进行。当在循环当中附加元素就会产生一个常见的问题。

$.each( myArray, function( i, item ) {
    var newListItem = "<li>" + item + "</li>";
    $( "#ballers" ).append( newListItem );
});

一个常用的技巧是利用文档片段(document fragment)。在循环的每一次迭代当中,将元素附加到片段而不是DOM元素当中。当循环结束后,将片段附加到DOM元素当中即可。

var frag = document.createDocumentFragment();
$.each( myArray, function( i, item ) {
    var newListItem = document.createElement( "li" );
    var itemText = document.createTextNode( item );
    newListItem.appendChild( itemText );
    frag.appendChild( newListItem );
});
$( "#ballers" )[ 0 ].appendChild( frag );

另一个简单的技巧是在循环的每次迭代当中,持续构建一个字符串。当循环结束后,将DOM元素的HTML设置成该字符串。

var myHtml = "";
$.each( myArray, function( i, item ) {
    myHtml += "<li>" + item + "</li>";
});
$( "#ballers" ).html( myHtml );

当然还有其它一些技巧可以供你尝试。一个名为 jsperf 的站点为测试这些性能提供了一条好的出路。该网站允许你使用基准测试每一个技巧,并将其跨平台的性能测试结果可视化的展现出来。

2. Cache Length During Loops

在for循环当中,不要每次都访问数组的 length 属性;应当事先将其缓存起来。

var myLength = myArray.length;
for ( var i = 0; i < myLength; i++ ) {
    // do stuff
}

3. Detach Elements to Work with Them

操作DOM是缓慢的,因此你想尽可能减少对齐进行操作。jQuery在1.4版本当中引入了名为 detach() 的方法来帮助解决这一问题,它允许你在对元素进行操作时,将它们从DOM当中分离出来。

var $table = $( "#myTable" );
var $parent = $table.parent();
$table.detach();
// ... add lots and lots of rows to table
$parent.append( $table );

4. Don't Act on Absent Elements

如果你正打算在一个空的选择器上运行大量的代码,jQuery并不会给予任何的提示 -- 它将会继续的执行,就像是没有发生任何的错误。必须由你来验证选择器包含了多少元素。

// Bad: This runs three functions before it
// realizes there's nothing in the selection
$( "#nosuchthing" ).slideUp();
// Better:
var $mySelection = $( "#nosuchthing" );
if ( $mySelection.length ) {
    $mySelection.slideUp();
}
// Best: Add a doOnce plugin.
jQuery.fn.doOnce = function( func ) {
    this.length && func.apply( this );
    return this;
}
$( "li.cartitems" ).doOnce(function() {

    // make it ajax! \o/

});

本指南特别适用于那些当选择器不包含元素时还需要大量的开销的 jQuery UI 部件。

5. Optimize Selectors

选择器的优化和过去比起来并不是那么的重要,因为很多浏览器都实现了 document.querySelectorAll() 方法并且jQuery将选择器的负担转移到了浏览器上面。但是仍然有一些技巧需要铭记在心。

基于ID的选择器

以一个ID作为选择器的开始总是最好的。

 // Fast:
 $( "#container div.robotarm" ); // Super-fast:
 $( "#container" ).find( "div.robotarm" );

采用 .find() 方法的方式将更加的快速,因为第一个选择器已经过处理,而无需通过嘈杂的选择器引擎 -- ID-Only的选择器已使用 document.getElementById() 方法进行处理,之所以快速,是因为它是浏览器的原生方法。

特异性

尽量详细的描述选择器的右侧,对于左侧则应反其道而行之。

 // Unoptimized:
 $( "div.data .gonzalez" ); // Optimized:
 $( ".data td.gonzalez" );

尽量在选择器的最右侧使用 tag.class 的形式来描述选择器,而在左侧则尽量只使用 tag 或者 .class 。

避免过度使用特异性

 $( ".data table.attendees td.gonzalez" ); // Better: Drop the middle if possible.
 $( ".data td.gonzalez" );

去讨好“DOM”总是有利于提升选择器的性能,因为选择器引擎在搜寻元素时无需进行太多的遍历。

避免使用通用选择器

如果一个选择器明确或暗示它能在不确定的范围内进行匹配将会大大影响性能。

 $( ".buttons > *" ); // Extremely expensive.
 $( ".buttons" ).children(); // Much better. $( ".category :radio" ); // Implied universal selection.
 $( ".category *:radio" ); // Same thing, explicit now.
 $( ".category input:radio" ); // Much better.
 Use Stylesheets for Changing CSS on Many Elements

假如你使用 .css() 方法来改变超过20个元素的CSS,应当考虑为页面添加一个样式标签作为替代,这样做可以提升将近60%的速度。

 // Fine for up to 20 elements, slow after that:
 $( "a.swedberg" ).css( "color", "#0769ad" ); // Much faster:
 $( "<style type=\"text/css\">a.swedberg { color: #0769ad }</style>")
     .appendTo( "head" );
 Don't Treat jQuery as a Black Box

把jQuery的源码当成文档,可以把它(http://bit.ly/jqsource)保存在你的收藏夹内,经常的查阅参考。

Javascript 相关文章推荐
jquery 指南/入门基础
Nov 30 Javascript
基于jQuery.Validate验证库知识点的详解
Apr 26 Javascript
iframe的父子窗口之间的对象相互调用基本用法
Sep 03 Javascript
原生javascript模仿win8等待提示圆圈进度条
Apr 24 Javascript
JavaScript利用正则表达式去除日期中的“-”
Jul 01 Javascript
js实现select二级联动下拉菜单
Apr 17 Javascript
[原创]jQuery常用的4种加载方式分析
Jul 25 Javascript
jq给页面添加覆盖层遮罩的实例
Feb 16 Javascript
深入分析element ScrollBar滚动组件源码
Jan 22 Javascript
Node.js学习之内置模块fs用法示例
Jan 22 Javascript
一篇文章带你使用Typescript封装一个Vue组件(简单易懂)
Jun 05 Javascript
JS如何实现封装列表右滑动删除收藏按钮
Jul 23 Javascript
为开发者准备的10款最好的jQuery日历插件
Feb 04 #Javascript
javascript简单性能问题及学习笔记
Feb 04 #Javascript
JavaScript初学者建议:不要去管浏览器兼容
Feb 04 #Javascript
21个值得收藏的Javascript技巧
Feb 04 #Javascript
Javascript 按位与运算符 (&amp;)使用介绍
Feb 04 #Javascript
Javascript 按位与赋值运算符 (&amp;=)使用介绍
Feb 04 #Javascript
Javascript 按位左移运算符使用介绍(
Feb 04 #Javascript
You might like
据说是雅虎的一份PHP面试题附答案
2009/01/07 PHP
php防止恶意刷新与刷票的方法
2014/11/21 PHP
PHP实现Snowflake生成分布式唯一ID的方法示例
2020/08/30 PHP
使用JavaScript动态设置样式实现代码(2)
2013/01/25 Javascript
jquery插件实现鼠标经过图片右侧显示大图的效果(类似淘宝)
2013/02/04 Javascript
自定义jQuery选项卡插件实例
2013/03/27 Javascript
JS图片切换的具体方法(带缩略图版)
2013/11/12 Javascript
JS中FRAME的操作问题实例分析
2014/10/21 Javascript
js实现简单的省市县三级联动效果实例
2016/02/18 Javascript
JQuery对ASP.NET MVC数据进行更新删除
2016/07/13 Javascript
如何用JS判断两个数字的大小
2016/07/21 Javascript
js 实现获取name 相同的页面元素并循环遍历的方法
2017/02/14 Javascript
node.js 中间件express-session使用详解
2017/05/20 Javascript
详解vue-cli 构建Vue项目遇到的坑
2017/08/30 Javascript
Bootstrap实现可折叠分组侧边导航菜单
2018/03/07 Javascript
JavaScript实现简单动态进度条效果
2018/04/06 Javascript
JavaScript多态与封装实例分析
2018/07/27 Javascript
如何在Vue中使用CleaveJS格式化你的输入内容
2018/12/14 Javascript
vue element动态渲染、移除表单并添加验证的实现
2019/01/16 Javascript
vue 地区选择器v-distpicker的常用功能
2019/07/23 Javascript
解决Vue在Tomcat8下部署页面不加载的问题
2019/11/12 Javascript
python实现调用其他python脚本的方法
2014/10/05 Python
Python功能键的读取方法
2015/05/28 Python
详解Python编程中基本的数学计算使用
2016/02/04 Python
Face++ API实现手势识别系统设计
2018/11/21 Python
python Selenium实现付费音乐批量下载的实现方法
2019/01/24 Python
python实现通过flask和前端进行数据收发
2019/08/22 Python
python argparser的具体使用
2019/11/10 Python
Python3 字典dictionary入门基础附实例
2020/02/10 Python
Python中的wordcloud库安装问题及解决方法
2020/05/27 Python
HealthElement海外旗舰店:新西兰大卖场
2018/02/23 全球购物
7 For All Mankind官网:美国加州洛杉矶的高级牛仔服装品牌
2018/12/20 全球购物
安全检查与奖惩制度
2014/01/23 职场文书
大学生英语演讲稿
2014/04/24 职场文书
2014矛盾纠纷排查调处工作总结
2014/12/09 职场文书
2016大学军训通讯稿
2015/11/25 职场文书