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 相关文章推荐
    js获取html参数及向swf传递参数应用介绍
    Feb 18 Javascript
    js借助ActiveXObject实现创建文件
    Sep 29 Javascript
    JavaScript使用focus()设置焦点失败的解决方法
    Sep 03 Javascript
    jQuery插件Tmpl的简单使用方法
    Apr 27 Javascript
    JavaScript实现点击按钮直接打印
    Jan 06 Javascript
    那些精彩的JavaScript代码片段
    Jan 12 Javascript
    angular 动态组件类型详解(四种组件类型)
    Feb 22 Javascript
    深入理解Vuex 模块化(module)
    Sep 26 Javascript
    JS实现返回上一页并刷新页面的方法分析
    Jul 16 Javascript
    vue+element 实现商城主题开发的示例代码
    Mar 26 Javascript
    Vue登录拦截 登录后继续跳转指定页面的操作
    Aug 04 Javascript
    npm全局环境变量配置详解
    Dec 15 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实现文件下载【实例分享】
    2017/04/28 PHP
    php pdo连接数据库操作示例
    2019/11/18 PHP
    JS代码格式化和语法着色V2
    2006/10/14 Javascript
    javascript基础之查找元素的详细介绍(访问节点)
    2013/07/05 Javascript
    微信WeixinJSBridge API使用实例
    2015/05/25 Javascript
    javascript中Array()数组函数详解
    2015/08/23 Javascript
    jQuery的选择器中的通配符[id^='code']或[name^='code']及jquery选择器总结
    2015/12/24 Javascript
    jQuery Dialog 打开时自动聚焦的解决方法(两种方法)
    2016/11/24 Javascript
    薪资那么高的Web前端必看书单
    2017/10/13 Javascript
    vue2.0 如何把子组件的数据传给父组件(推荐)
    2018/01/15 Javascript
    jQuery实现table表格checkbox全选的方法分析
    2018/07/04 jQuery
    node.js使用免费的阿里云ip查询获取ip所在地【推荐】
    2018/09/03 Javascript
    原生JS实现的放大镜特效示例【测试可用】
    2018/12/08 Javascript
    vue全局自定义指令-元素拖拽的实现代码
    2019/04/14 Javascript
    小程序如何获取多个formId实现详解
    2019/09/20 Javascript
    微信小程序使用 vant Dialog组件的正确方式
    2020/02/21 Javascript
    vue实现一个6个输入框的验证码输入组件功能的实例代码
    2020/06/29 Javascript
    angular共享依赖的解决方案分享
    2020/10/15 Javascript
    python脚本监控docker容器
    2016/04/27 Python
    python中星号变量的几种特殊用法
    2016/09/07 Python
    PyTorch读取Cifar数据集并显示图片的实例讲解
    2018/07/27 Python
    Python选择网卡发包及接收数据包
    2019/04/04 Python
    使用Python将图片转正方形的两种方法实例代码详解
    2020/04/29 Python
    keras 指定程序在某块卡上训练实例
    2020/06/22 Python
    Pycharm 2020.1 版配置优化的详细教程
    2020/08/07 Python
    eBay澳大利亚站:eBay.com.au
    2018/02/02 全球购物
    西班牙宠物用品和食品网上商店:Tiendanimal
    2019/06/06 全球购物
    Paper Cape官网:美国婴儿和儿童服装品牌
    2019/11/02 全球购物
    高中毕业生个人自我鉴定
    2013/11/24 职场文书
    简单而又朴实的个人求职信分享
    2013/12/12 职场文书
    仓库理货员岗位职责
    2013/12/18 职场文书
    质量承诺书格式
    2014/05/20 职场文书
    团队精神口号
    2014/06/06 职场文书
    禁烟标语大全
    2014/06/11 职场文书
    党纪处分决定书
    2015/06/24 职场文书
    Go语言实现Base64、Base58编码与解码
    2021/07/26 Golang