jQuery 性能优化手册 推荐


Posted in Javascript onFebruary 23, 2010

在twitter上发现了<jQuery Performance Rules>这篇文章, 简单的摘译了一下:

总是从ID选择器开始继承
在class前使用tag
将jquery对象缓存起来
掌握强大的链式操作
使用子查询
对直接的DOM操作进行限制
冒泡
消除无效查询
推迟到 $(window).load
压缩js
全面掌握jquery库
1. 总是从ID选择器开始继承
在jquery中最快的选择器是ID选择器. 因为它直接来自于Javascript的getElementById()方法.

<div id=“content”> 
<form method=“post” action=“/”> 
<h2>Traffic Light</h2> 
<ul id=“traffic_light”> 
<li><input type=“radio” class=“on” name=“light” value=“red” /> Red</li> 
<li><input type=“radio” class=“off” name=“light” value=“yellow” /> Yellow</li> 
<li><input type=“radio” class=“off” name=“light” value=“green” /> Green</li> 
</ul> 
<input class=“button” id=“traffic_button” type=“submit” value=“Go” /> 
</form> 
</div>

像这样选择按钮是低效的:
var traffic_button = $(‘#content .button');

用ID直接选择按钮效率更高:
var traffic_button = $(‘#traffic_button');

选择多个元素

提到多元素选择其实是在说DOM遍历和循环, 这些都是比较慢的东西.为了提高性能, 最好从就近的ID开始继承.

var traffic_lights = $(‘#traffic_light input');

2. 在class前使用tag

第二快的选择器是tag选择器($('head')). 同理,因为它来自原生的getElementsByTagName() 方法.

<div id=“content”> 
<form method=“post” action=“/”> 
<h2>Traffic Light</h2> 
<ul id=“traffic_light”> 
<li><input type=“radio” class=“on” name=“light” value=“red” /> Red</li> 
<li><input type=“radio” class=“off” name=“light” value=“yellow” /> Yellow</li> 
<li><input type=“radio” class=“off” name=“light” value=“green” /> Green</li> 
</ul> 
<input class=“button” id=“traffic_button” type=“submit” value=“Go” /> 
</form> 
</div>

总是用一个tag name来限制(修饰)class (并且不要忘记就近的ID):
var active_light = $(‘#traffic_light input.on');

注意: 在jquery中Class是最慢的选择器. IE浏览器下它会遍历所有DOM节点不管它用在那里.

不要用用tag name来修饰ID. 下面的例子将会遍历所有的div元素来查找id为'content'的哪一个节点:

var content = $(‘div#content');

用ID修饰ID也是画蛇添足:
var traffic_light = $(‘#content #traffic_light');

3.将jquery对象缓存起来

要养成将jquery对象缓存进变量的习惯.

永远不要这样做:

$(‘#traffic_light input.on).bind('click‘, function(){…}); 
$('#traffic_light input.on).css(‘border', ‘3px dashed yellow'); 
$(‘#traffic_light input.on).css('background-color‘, ‘orange‘); 
$('#traffic_light input.on).fadeIn('slow');

最好先将对象缓存进一个变量然后再操作:
var $active_light = $(‘#traffic_light input.on'); 
$active_light.bind(‘click', function(){…}); 
$active_light.css(‘border', ‘3px dashed yellow'); 
$active_light.css(‘background-color', ‘orange'); 
$active_light.fadeIn('slow');

为了记住我们本地变量是jquery的封装, 通常用一个$作为变量前缀. 记住,永远不要让相同的选择器在你的代码里出现多次.

缓存jquery结果,备用

如果你打算将jquery结果对象用在程序的其它部分,或者你的function会多次执行, 那么就将他们缓存到一个全局变量中.

定义一个全局容器来存放jquery结果, 我们就可以在其它函数引用它们:

// 在全局范围定义一个对象 (例如: window对象) 
window.$my = 
{ 
// 初始化所有可能会不止一次要使用的查询 
head : $(‘head'), 
traffic_light : $(‘#traffic_light'), 
traffic_button : $(‘#traffic_button') 
}; function do_something() 
{ 
// 现在你可以引用存储的结果并操作它们 
var script = document.createElement('script'); 
$my.head.append(script); 
// 当你在函数内部操作是, 可以继续将查询存入全局对象中去. 
$my.cool_results = $(‘#some_ul li'); 
$my.other_results = $(‘#some_table td'); 
// 将全局函数作为一个普通的jquery对象去使用. 
$my.other_results.css(‘border-color', ‘red'); 
$my.traffic_light.css(‘border-color', ‘green'); 
}

4. 掌握强大的链式操作

上面的例子也可以写成这样:

var $active_light = $(‘#traffic_light input.on');$active_light.bind(‘click', function(){…}) 
.css(‘border', ‘3px dashed yellow') 
.css(‘background-color', ‘orange') 
.fadeIn('slow');

这样可以写更少的代码, 让我们的js更轻量.

5.使用子查询

jQuery 允许我们对一个已包装的对象使用附加的选择器操作. 因为我们已经在保存了一个父级对象在变量里, 这样大大提高对其子元素的操作:

<div id=“content”> 
<form method=“post” action=“/”> 
<h2>Traffic Light</h2> 
<ul id=“traffic_light”> 
<li><input type=“radio” class=“on” name=“light” value=“red” /> Red</li> 
<li><input type=“radio” class=“off” name=“light” value=“yellow” /> Yellow</li> 
<li><input type=“radio” class=“off” name=“light” value=“green” /> Green</li> 
</ul> 
<input class=“button” id=“traffic_button” type=“submit” value=“Go” /> 
</form> 
</div>

例如, 我们可以用子查询的方法来抓取到亮或不亮的灯, 并缓存起来以备后续操作.
var $traffic_light = $(‘#traffic_light'), 
$active_light = $traffic_light.find(‘input.on'), 
$inactive_lights = $traffic_light.find(‘input.off');

提示: 你可以用逗号分隔的方法一次声明多个局部变量?节省字节数

6.对直接的DOM操作进行限制

这里的基本思想是在内存中建立你确实想要的东西,然后更新DOM 。这并不是一个jQuery最佳实践,但必须进行有效的JavaScript操作 。直接的DOM操作速度很慢。

例如,你想动态的创建一组列表元素, 千万不要这么做:

var top_100_list = [...], // 假设这里是100个独一无二的字符串 
$mylist = $(‘#mylist'); // jQuery 选择到 <ul> 元素for (var i=0, l=top_100_list.length; i<l; i++) 
{ 
$mylist.append(‘<li>' + top_100_list[i] + ‘</li>'); 
}

我们应该将整套元素字符串在插入进dom中之前全部创建好:
var top_100_list = [...], 
$mylist = $(‘#mylist'), 
top_100_li = “”; // 这个变量将用来存储我们的列表元素for (var i=0, l=top_100_list.length; i<l; i++) 
{ 
top_100_li += ‘<li>' + top_100_list[i] + ‘</li>'; 
} 
$mylist.html(top_100_li);

我们在插入之前将多个元素包裹进一个单独的父级节点会更快:
var top_100_list = [...], 
$mylist = $(‘#mylist'), 
top_100_ul = ‘<ul id=”#mylist”>';for (var i=0, l=top_100_list.length; i<l; i++) 
{ 
top_100_ul += ‘<li>' + top_100_list[i] + ‘</li>'; 
} 
top_100_ul += ‘</ul>'; //关闭无序列表 
$mylist.replaceWith(top_100_ul);

如果你做了以上几条还是担心有性能问题,那么:
  • 试试jquery的 clone() 方法, 它会创建一个节点树的副本, 它允许以”离线”的方式进行dom操作, 当你操作完成后再将其放回到节点树里.
  • 使用 DOM DocumentFragments. 正如jQuery作者所言, 它的性能要明显优于直接的dom操作.
    7. 冒泡

    除非在特殊情况下, 否则每一个js事件(例如:click, mouseover, 等.)都会冒泡到父级节点. 当我们需要给多个元素调用同个函数时这点会很有用.

    代替这种效率很差的多元素事件监听的方法就是, 你只需向它们的父节点绑定一次, 并且可以计算出哪个节点触发了事件.

    例如, 我们要为一个拥有很多输入框的表单绑定这样的行为: 当输入框被选中时为它添加一个class

    像这样绑定事件是低效的:

    $(‘#entryform input).bind('focus‘, function(){ 
    $(this).addClass('selected‘); 
    }).bind('blur‘, function(){ 
    $(this).removeClass('selected‘); 
    });

    我们需要在父级监听获取焦点和失去焦点的事件:
    $(‘#entryform').bind(‘focus', function(e){ 
    var cell = $(e.target); // e.target grabs the node that triggered the event. 
    cell.addClass('selected'); 
    }).bind(‘blur', function(e){ 
    var cell = $(e.target); 
    cell.removeClass('selected'); 
    });

    父级元素扮演了一个调度员的角色, 它可以基于目标元素绑定事件. 如果你发现你给很多元素绑定了同一个事件监听, 那么你肯定哪里做错了.
    8.消除无效查询
    尽管jquery可以很优雅的处理没有匹配元素的情况, 但它还是需要花费时间去寻找. 如果你整站只有一个全局js, 那么极有可能把所有的jquery函数塞进$(document)ready(function(){//所有你引以为傲的代码})里.
    只运行在页面里用到的函数. 大多数有效的方法就是使用行内初始化函数, 这样你的模板就能准确的控制何时何处该执行js.
    例如, 你的”文章”页面模板, 你可能会引用如下的代码在body结束处:
    <script type=“text/javascript> 
    mylib.article.init(); 
    </script> 
    </body>

    如果你的页面模板包含一些多变的模块可能不会出现在页面中, 或者为了视觉呈现的原因你需要它们能够快速加载, 你可以将初始化函数紧跟在模块之后.
    <ul id=“traffic_light”> 
    <li><input type=“radio” class=“on” name=“light” value=“red” /> Red</li> 
    <li><input type=“radio” class=“off” name=“light” value=“yellow” /> Yellow</li> 
    <li><input type=“radio” class=“off” name=“light” value=“green” /> Green</li> 
    </ul> 
    <script type=“text/javascript> 
    mylib.traffic_light.init(); 
    </script>

    你的全局js库可能会是这样子的:
    var mylib = 
    { 
    article_page : 
    { 
    init : function() 
    { 
    // Article 特有的jQuery函数. 
    } 
    }, 
    traffic_light : 
    { 
    init : function() 
    { 
    // Traffic light 特有的jQuery函数. 
    } 
    } 
    }

    9. 推迟到 $(window).load

    jquery对于开发者来说有一个很诱人的东西, 可以把任何东西挂到$(document).ready下冒充事件. 在大多数例子中你都会发现这样的情况.

    尽管$(document).rady 确实很有用, 它可以在页面渲染时,其它元素还没下载完成就执行. 如果你发现你的页面一直是载入中的状态, 很有可能就是$(document).ready函数引起的.

    你可以通过将jquery函数绑定到$(window).load 事件的方法来减少页面载入时的cpu使用率. 它会在所有的html(包括<iframe>)被下载完成后执行.

    $(window).load(function(){ 
    // 页面完全载入后才初始化的jQuery函数. 
    });

    多余的功能例如拖放, 视觉特效和动画, 预载入隐藏图像,等等. 都是适合这种技术的场合.

    10. 压缩js

    推荐一个js在线压缩地址:
    http://dean.edwards.name/packer/
    https://3water.com/tools/packer.htm
    11. 全面掌握jquery库

    知己知彼, 百战百胜. 只有更深入的了解jQuery才能更灵活的使用它. 这里提供一个jQuery的速查手册, 可以打印出来随身携带. 要是有能力将jQuery源码通读一遍那就更好了.
    文来自:jQuery Performance Rules ; 译文来自:Rlog.cn . 若转载请注明出处, 谢谢

  • Javascript 相关文章推荐
    深入聊聊Array的sort方法的使用技巧.详细点评protype.js中的sortBy方法
    Apr 12 Javascript
    基于jQuery的左右滚动实现代码
    Dec 03 Javascript
    javascript中数组的定义及使用实例
    Jan 21 Javascript
    js实现无缝滚动特效
    Dec 20 Javascript
    vue.js学习笔记之绑定style样式和class列表
    Oct 31 Javascript
    js 点击a标签 获取a的自定义属性方法
    Nov 21 Javascript
    原生Javascript插件开发实践
    Jan 09 Javascript
    javascript实现滑动解锁功能
    Mar 22 Javascript
    基于Cookie常用操作以及属性介绍
    Sep 07 Javascript
    Angular 4.0学习教程之架构详解
    Sep 12 Javascript
    vue中当图片地址无效的时候,显示默认图片的方法
    Sep 18 Javascript
    Vue项目接入Paypal实现示例详解
    Jun 04 Javascript
    jquery实现的超出屏幕时把固定层变为定位层的代码
    Feb 23 #Javascript
    JSON 学习之JSON in JavaScript详细使用说明
    Feb 23 #Javascript
    js下用层来实现select的title提示属性
    Feb 23 #Javascript
    jquery 锁定弹出层实现代码
    Feb 23 #Javascript
    javascript document.compatMode兼容性
    Feb 23 #Javascript
    js操作ajax返回的json的注意问题!
    Feb 23 #Javascript
    javascript入门基础之私有变量
    Feb 23 #Javascript
    You might like
    php生成WAP页面
    2006/10/09 PHP
    php基于SQLite实现的分页功能示例
    2017/06/21 PHP
    jQuery EasyUI API 中文文档 - Tabs标签页/选项卡
    2011/10/01 Javascript
    js replace正则表达式应用案例讲解
    2013/01/17 Javascript
    Nodejs极简入门教程(二):定时器
    2014/10/25 NodeJs
    jQuery中Ajax的load方法详解
    2015/01/14 Javascript
    JavaScript 学习笔记之变量及其作用域
    2015/01/14 Javascript
    JS显示表格内指定行html代码的方法
    2015/03/31 Javascript
    jQuery Jsonp跨域模拟搜索引擎
    2017/06/17 jQuery
    Angular中ng-options下拉数据默认值的设定方法
    2017/06/21 Javascript
    Bootstrap 3多级下拉菜单实例
    2017/11/23 Javascript
    JavaScript中call和apply方法的区别实例分析
    2018/08/03 Javascript
    详解Vue CLI3配置解析之css.extract
    2018/09/14 Javascript
    Vue-CLI 项目在pycharm中配置方法
    2019/08/30 Javascript
    实例分析JS中的相等性判断===、 ==和Object.is()
    2019/11/17 Javascript
    node.js事件轮询机制原理知识点
    2019/12/22 Javascript
    原生js生成图片验证码
    2020/10/11 Javascript
    [44:40]KG vs LGD 2019国际邀请赛小组赛 BO2 第一场 8.15
    2019/08/16 DOTA
    Python找出文件中使用率最高的汉字实例详解
    2015/06/03 Python
    Python 3.6 读取并操作文件内容的实例
    2018/04/23 Python
    Python异常处理操作实例详解
    2018/05/10 Python
    解决python中使用plot画图,图不显示的问题
    2018/07/04 Python
    对python 数据处理中的LabelEncoder 和 OneHotEncoder详解
    2018/07/11 Python
    python多线程分块读取文件
    2019/08/29 Python
    Python实现验证码识别
    2020/06/15 Python
    Python sklearn中的.fit与.predict的用法说明
    2020/06/28 Python
    PyQt中使用QtSql连接MySql数据库的方法
    2020/07/28 Python
    Python在线和离线安装第三方库的方法
    2020/10/31 Python
    html5 localStorage本地存储_动力节点Java学院整理
    2017/07/06 HTML / CSS
    Html5移动端弹幕动画实现示例代码
    2018/08/27 HTML / CSS
    西班牙英格列斯百货法国官网:El Corte Inglés法国
    2017/07/09 全球购物
    竞聘书模板
    2014/03/31 职场文书
    2014年预算员工作总结
    2014/12/05 职场文书
    2019年工作总结范文
    2019/05/21 职场文书
    pytorch 运行一段时间后出现GPU OOM的问题
    2021/06/02 Python
    Python代码风格与编程习惯重要吗?
    2021/06/03 Python