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 相关文章推荐
    jquery+json实现的搜索加分页效果
    Mar 31 Javascript
    jQuery源码中的chunker 正则过滤符分析
    Jul 31 Javascript
    javascript 四十条常用技巧大全
    Sep 09 Javascript
    jQuery动态生成Bootstrap表格
    Nov 01 Javascript
    教你一步步用jQyery实现轮播器
    Dec 18 Javascript
    详解Vue使用 vue-cli 搭建项目
    Apr 20 Javascript
    xmlplus组件设计系列之分隔框(DividedBox)(8)
    May 02 Javascript
    vue组件发布到npm简单步骤
    Nov 30 Javascript
    jQuery实现带右侧索引功能的通讯录示例【附源码下载】
    Apr 17 jQuery
    微信小程序登录换取token的教程
    May 31 Javascript
    Vue computed 计算属性代码实例
    Apr 22 Javascript
    详解关于Vue单元测试的几个坑
    Apr 26 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写的简易聊天室代码
    2011/06/04 PHP
    PHP APC缓存配置、使用详解
    2014/03/06 PHP
    PHP创建PowerPoint2007文档的方法
    2015/12/10 PHP
    深入理解PHP类的自动载入机制
    2016/09/16 PHP
    浅谈htmlentities 、htmlspecialchars、addslashes的使用方法
    2016/12/09 PHP
    PHP基于DOMDocument解析和生成xml的方法分析
    2017/07/17 PHP
    PHP+Apache实现二级域名之间共享cookie的方法
    2019/07/24 PHP
    javascript下给元素添加事件的方法与代码
    2007/08/13 Javascript
    JQuery AJAX提交中文乱码的解决方案
    2010/07/02 Javascript
    jquery对元素拖动排序示例
    2014/01/16 Javascript
    jquery根据锚点offset值实现动画切换
    2014/09/11 Javascript
    thinkphp实现无限分类(使用递归)
    2015/12/19 Javascript
    JavaScript禁止微信浏览器下拉回弹效果
    2017/05/16 Javascript
    layui2.0使用table+laypage实现真分页
    2019/07/27 Javascript
    vue或react项目生产环境去掉console.log的操作
    2020/09/02 Javascript
    微信小程序实现打卡签到页面
    2020/09/21 Javascript
    react项目从新建到部署的实现示例
    2021/02/19 Javascript
    [43:41]OG vs Newbee 2019国际邀请赛淘汰赛 胜者组 BO3 第一场 8.21.mp4
    2020/07/19 DOTA
    采用python实现简单QQ单用户机器人的方法
    2014/07/03 Python
    Python的for和break循环结构中使用else语句的技巧
    2016/05/24 Python
    利用Python开发实现简单的记事本
    2016/11/15 Python
    基于python socketserver框架全面解析
    2017/09/21 Python
    Django REST framework 分页的实现代码
    2019/06/19 Python
    如何使用python操作vmware
    2019/07/27 Python
    python 实现字符串下标的输出功能
    2020/02/13 Python
    Python连接Hadoop数据中遇到的各种坑(汇总)
    2020/04/14 Python
    HTML5的一个显示电池状态的API简介
    2015/06/18 HTML / CSS
    PHP如何自定义函数
    2016/09/16 面试题
    有针对性的求职自荐信
    2013/11/14 职场文书
    六年级学生评语大全
    2014/12/26 职场文书
    离职感谢信
    2015/01/21 职场文书
    清洁工个人总结
    2015/03/04 职场文书
    学术研讨会主持词
    2015/07/04 职场文书
    一次SQL如何查重及去重的实战记录
    2022/03/13 MySQL
    virtualenv隔离Python环境的问题解析
    2022/06/21 Python
    Java代码规范与质量检测插件SonarLint的使用
    2022/08/05 Java/Android