jquery提升性能最佳实践小结


Posted in Javascript onDecember 06, 2010

将jquery对象缓存起来在
for循环中,不要每次都要访问数组的length属性,我们应该先将对象缓存进一个变量然后再操作,如下所示:

var myLength = myArray.length; 
for (var i = 0; i < myLength; i++) { 
// 要做的事 
}

在循环外使用append 进行DOM操作是有代价的,如果需要往DOM中添加大量元素,你应该一次批量完成,而不是一次一个。
// 别这样 
$.each(reallyLongArray, function(count, item) { 
var newLI = '<li>' + item + '</li>'; 
$('#ballers').append(newLI); 
}); 
//较好的做法 
var frag = document.createDocumentFragment(); 
$.each(reallyLongArray, function(count, item) { 
var newLI = '<li>' + item + '</li>'; 
frag.appendChild(newLI[0]); 
}); 
$('#ballers')[0].appendChild(frag);不要在each()里用$('#id')的选择器,会有多次遍历查找dom元素,效率极低用document.createDocumentFragment()来减少页面的DOM结构改变的次数、刷新的次数 // 或者这样 
var myHtml = ''; 
$.each(myArray, function(i, item) { 
html += '<li>' + item + '</li>'; 
}); 
$('#ballers').html(myHtml);

保持简洁风格
// 不理想 
if ($ventfade.data('currently') != 'showing') { 
$ventfade.stop(); 
} 
if ($venthover.data('currently') != 'showing') { 
$venthover.stop(); 
} 
if ($spans.data('currently') != 'showing') { 
$spans.stop(); 
} 
// 较好的 
var elems = [$ventfade, $venthover, $spans]; 
$.each(elems, function(k, v) { 
if (v.data('currently') != 'showing') { 
v.stop(); 
} 
})

慎用匿名函数
匿名函数的约束到处都是一种痛苦。他们难以调试,维护,测试或重用。相反,我们可以使用对象封装,将那些处理和回调函数组织并通过命名管理起来。
// 不要这样 
$(document).ready(function() {... 
$('#magic').click(function(e) { 
$('#yayeffects').slideUp(function() {... 
}); 
}); 
$('#happiness').load(url + ' #unicorns', function() {... 
}) 
}); // 较好的 
var PI = { 
onReady: function() {... 
$('#magic').click(PI.candyMtn); 
$('#happiness').load(url + ' #unicorns', PI.unicornCb); 
}, 
candyMtn: function(e) { 
$('#yayeffects').slideUp(PI.slideCb); 
}, 
slideCb: function() {... 
}, 
unicornCb: function() {... 
} 
} 
$(document).ready(PI.onReady);

优化选择器
节点选择和DOM操作, 根据给定的ID匹配一个元素总是使用#id去寻找element
// 非常快 
$('#container div.robotarm'); 
// 超级快 
$('#container').find('div.robotarm');使用$.fn.find方法更快一些,因为第一个选择器是无须经过选择器引擎处理,在jquery中最快的选择器是ID选择器.因为它直接来自于Javascript的getElementById()方法,这是非常快,因为它是原产于浏览器。如果你需要选择多个元素,这必然会涉及到DOM遍历和循环,为了提高性能,建议从最近的ID开始继承。

具体指定选择器的右侧部分应该尽可能具体,左侧则不需要那么具体。
// 未优化 
$('div.data .gonzalez'); 
// 优化后 
$('.data td.gonzalez');如果可以,尽量在选择器靠右侧的部分使用tag.class,而左侧只有tag或者只有.class。

避免过度的具体
$('.data table.attendees td.gonzalez'); 
// 不写中间的会更好 
$('.data td.gonzalez');清爽的DOM结构也有助于改善选择器的性能,选择器引擎可以少跑几层去寻觅那个元素了。

避免使用无定向通配符选择器

$('.buttons > *'); // 极慢 
$('.buttons').children(); // 快很多 
$('.gender :radio'); // 无定向搜索 
$('.gender *:radio'); // 同上 
$('.gender input:radio'); // 这样 好很多

使用事件委派

事件委派允许你为一个容器元素(例如,一个无序列表)绑定一个事件处理程序,而不需给容器内每个元素(例如,列表项)都去绑定。jQuery提供$.fn.live和$.fn.delegate。如果可能的话,你应该使用$.fn.delegate而不是$.fn.live,因为它省去了不必要的选择需要,其明确的情况下(对$.fn.live的文档的上下文),减少了大约80 % 的开销。除了有性能提升的好处,事件委派也使你在往容器里添加新元素时无需重新绑定事件,因为已经有了。

通过事件委派一次绑定多种事件,以减少事件冗余

// 不好的 (如果列表里面元素很多) 
$('li.trigger').click(handlerFn); // 较好的: event delegation with $.fn.live 
$('li.trigger').live('click', handlerFn); 
// 最优的: $.fn.delegate 
$('#myList').delegate('li.trigger', 'click', handlerFn);

移除元素
DOM操作是慢的,你应该尽量避免去操作它。jQuery在1.4版引入了
$.fn.detach从DOM中去掉所有匹配的元素。
var $table = $('#myTable'); 
var $parent = table.parent(); 
$table.detach(); 
// ... 添加大量的行到表格中 
$parent.append(table);

.detach()和.remove()一样, 除了.detach()保存所有jQuery数据和被移走的元素相关联。当需要移走一个元素,不久又将该元素插入DOM时,这种方法很有用。

当大量元素修改CSS,应该使用样式表

如果你在用$.fn.css给多于20个元素修改CSS,考虑一下添加一个style标签,这样可以速度可以提升60 % 。

// 多于20个 明显慢 
$('a.swedberg').css('color', '#asd123'); 
$('<style type="text/css">a.swedberg { color : #asd123 }</style>').appendTo('head');

使用$.data而不是$.fn.data将$.data应用于DOM元素比直接调用jQuery选择结果的$.fn.data要快上10倍.虽然,这要先确定你是理解DOM元素与jQuery选择结果之间的区别的。
// 常用 
$(elem).data(key, value); 
// 快十倍 
$.data(elem, key, value);

别费时间在空白的选择结果上了

jQuery将不会告诉你,如果你想运行的代码在一个空选择上,它会继续运行,好像没有什么错。影响性能。

//太遭了,执行了三个方法后才意识到里面是空的 
$('#nosuchthing').slideUp(); // 较好 
var $mySelection = $('#nosuchthing'); 
if ($mySelection.length) { 
mySelection.slideUp(); 
} 
// 最佳: 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 widget,因为即使操作的元素为空其开销也不少。

定义变量
变量可以定义一个声明而不是几个

// 老套写法 
var test = 1; 
var test2 = function() {... 
}; 
var test3 = test2(test); // 新 
var test = 1, 
test2 = function() {... 
}, 
test3 = test2(test);在自动执行的函数,变量的定义可以完全省掉。 (function(foo, bar) {... 
})(1, 2);

条件判断

// 旧方法 
if (type == 'foo' || type == 'bar') {... 
} // 好方法 
if (/^(foo|bar)$/.test(type)) {... 
} 
// 查找对象 
if (({ 
foo: 1, 
bar: 1 
})[type]) {... 
}

作者:曾祥展
出处:学无止境 (http://www.cnblogs.com/zengxiangzhan/)
Javascript 相关文章推荐
Javascript中的数据类型之旅
Oct 18 Javascript
javascript实现加载xml文件的方法
Nov 24 Javascript
基于BootStarp的Dailog
Apr 28 Javascript
JS事件添加和移出的兼容写法示例
Jun 20 Javascript
KnockoutJS 3.X API 第四章之表单value绑定
Oct 10 Javascript
jQuery实现的小图列表,大图展示效果幻灯片示例
Oct 25 Javascript
jQuery内容筛选选择器实例代码
Feb 06 Javascript
JS+HTML5 FileReader对象用法示例
Apr 07 Javascript
详解jquery插件jquery.viewport.js学习使用方法
Sep 08 jQuery
zTree jQuery 树插件的使用(实例讲解)
Sep 25 jQuery
关于React动态加载路由处理的相关问题
Jan 07 Javascript
JS使用JSON.parse(),JSON.stringify()实现对对象的深拷贝功能分析
Mar 06 Javascript
Colortip基于jquery的信息提示框插件在IE6下面的显示问题修正方法
Dec 06 #Javascript
使用JS进行目录上传(相当于批量上传)
Dec 05 #Javascript
js对象的构造和继承实现代码
Dec 05 #Javascript
收集的一些Array及String原型对象的扩展实现代码
Dec 05 #Javascript
script标签的 charset 属性使用说明
Dec 04 #Javascript
本地对象Array的原型扩展实现代码
Dec 04 #Javascript
悄悄用脚本检查你访问过哪些网站的代码
Dec 04 #Javascript
You might like
PHP语言中global和$GLOBALS[]的分析 之二
2012/02/02 PHP
利用phpExcel实现Excel数据的导入导出(全步骤详细解析)
2013/11/26 PHP
PHP SplObjectStorage使用实例
2015/05/12 PHP
php使用正则表达式去掉html中的注释方法
2016/11/03 PHP
PHP5.0~5.6 各版本兼容性cURL文件上传功能实例分析
2018/05/11 PHP
JQuery中根据属性或属性值获得元素(6种情况获取方法)
2013/01/17 Javascript
JS格式化数字金额用逗号隔开保留两位小数
2013/10/18 Javascript
使用javascript做的一个随机点名程序
2014/02/13 Javascript
关于JavaScript对象的动态选择及遍历对象
2014/03/10 Javascript
js使用正则实现ReplaceAll全部替换的方法
2014/08/22 Javascript
js实现支持手机滑动切换的轮播图片效果实例
2015/04/29 Javascript
javascript中return,return true,return false三者的用法及区别
2015/11/17 Javascript
Bootstrap模态框(Modal)实现过渡效果
2017/03/17 Javascript
从对象列表中获取一个对象的方法,依据关键字和值
2017/09/20 Javascript
angular2中使用第三方js库的实例
2018/02/26 Javascript
vuejs项目打包之后的首屏加载优化及打包之后出现的问题
2018/04/01 Javascript
javascript中的相等操作符(==与===区别)
2019/12/21 Javascript
实例讲解React 组件
2020/07/07 Javascript
VSCode插件安装完成后的配置(常用配置)
2020/08/24 Javascript
详解Python程序与服务器连接的WSGI接口
2015/04/29 Python
Flask-Mail用法实例分析
2018/07/21 Python
python实现猜拳小游戏
2020/04/05 Python
使用Python和百度语音识别生成视频字幕的实现
2020/04/09 Python
python实现mask矩阵示例(根据列表所给元素)
2020/07/30 Python
Python2及Python3如何实现兼容切换
2020/09/01 Python
CSS3实现可爱的小黄人动画
2016/07/11 HTML / CSS
ProBikeKit美国官网:自行车套件,跑步和铁人三项套件
2016/10/13 全球购物
lookfantastic荷兰:在线购买奢华护肤、护发和化妆品
2018/11/27 全球购物
办公室秘书岗位职责范本
2014/02/11 职场文书
鸿星尔克广告词
2014/03/21 职场文书
升学宴主持词
2014/04/02 职场文书
小学社团活动总结
2014/06/27 职场文书
森马旗舰店双十一营销方案
2014/09/29 职场文书
干部职工纪律作风整改措施思想汇报
2014/10/11 职场文书
个人工作表现自我评价
2015/03/06 职场文书
2016年秋季运动会广播稿
2015/12/21 职场文书