优化Jquery,提升网页加载速度


Posted in Javascript onNovember 14, 2013

总是从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函数.
});

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

Javascript 相关文章推荐
JS获得浏览器版本和操作系统版本的例子
May 13 Javascript
浅谈Jquery为元素绑定事件
Apr 27 Javascript
javascript中 try catch用法
Aug 16 Javascript
漂亮! js实现颜色渐变效果
Aug 12 Javascript
jquery判断类型是不是number类型的实例代码
Oct 07 Javascript
微信和qq时间格式模板实例详解
Oct 21 Javascript
Extjs gridpanel 中的checkbox(复选框)根据某行的条件不能选中的解决方法
Feb 17 Javascript
基于Bootstrap框架实现图片切换
Mar 10 Javascript
从0到1构建vueSSR项目之node以及vue-cli3的配置
Mar 07 Javascript
帮你彻底搞懂JS中的prototype、__proto__与constructor(图解)
Aug 23 Javascript
element表格翻页第2页从1开始编号(后端从0开始分页)
Dec 10 Javascript
JS寄快递地址智能解析的实现代码
Jul 16 Javascript
Jquery AJAX POST与GET之间的区别
Nov 14 #Javascript
Jquery中$.get(),$.post(),$.ajax(),$.getJSON()的用法总结
Nov 14 #Javascript
JS正则表达式大全(整理详细且实用)
Nov 14 #Javascript
JS中typeof与instanceof之间的区别总结
Nov 14 #Javascript
jQuery探测位置的提示弹窗(toolTip box)详细解析
Nov 14 #Javascript
使用jQuery解决IE与FireFox下createElement方法的差异
Nov 14 #Javascript
浅析用prototype定义自己的方法
Nov 14 #Javascript
You might like
php读取数据库信息的几种方法
2008/05/24 PHP
php 缓存函数代码
2008/08/27 PHP
PHP反射使用实例和PHP反射API的中文说明
2014/07/02 PHP
Yii基于数组和对象的Model查询技巧实例详解
2015/12/28 PHP
jQuery find和children方法使用
2011/01/31 Javascript
基于jquery的放大镜效果
2012/05/30 Javascript
Javascript在IE和FireFox中的不同表现简析
2012/12/03 Javascript
Bootstrap多级导航栏(级联导航)的实现代码
2016/03/08 Javascript
NodeJs——入门必看攻略
2016/06/27 NodeJs
seajs学习之模块的依赖加载及模块API的导出
2016/10/20 Javascript
JS实现PC手机端和嵌入式滑动拼图验证码三种效果
2017/02/15 Javascript
angular ng-click防止重复提交实例
2017/06/16 Javascript
基于js粘贴事件paste简单解析以及遇到的坑
2017/09/07 Javascript
javascript性能优化之分时函数的介绍
2018/03/28 Javascript
详解element-ui日期时间选择器的日期格式化问题
2019/04/08 Javascript
微信小程序实现页面跳转传递参数(实体,对象)
2019/08/12 Javascript
vue集成chart.js的实现方法
2019/08/20 Javascript
js判断鼠标移入移出方向的方法
2020/06/24 Javascript
[26:40]DOTA2上海特级锦标赛A组资格赛#1 Secret VS MVP.Phx第一局
2016/02/25 DOTA
Python 文件和输入输出小结
2013/10/09 Python
Python函数式编程指南(四):生成器详解
2015/06/24 Python
使用Python的urllib2模块处理url和图片的技巧两则
2016/02/18 Python
python 随机森林算法及其优化详解
2019/07/11 Python
解决pycharm不能自动补全第三方库的函数和属性问题
2020/03/12 Python
matplotlib阶梯图的实现(step())
2021/03/02 Python
利用CSS3的flexbox实现水平垂直居中与三列等高布局
2016/09/12 HTML / CSS
html5实现的便签特效(实战分享)
2013/11/29 HTML / CSS
iphoneX 适配客户端H5页面的方法教程
2017/12/08 HTML / CSS
工会换届选举方案
2014/05/21 职场文书
党旗在我心中演讲稿
2014/09/15 职场文书
2014小学语文教学工作总结
2014/12/17 职场文书
家属慰问信
2015/02/14 职场文书
2015年家长学校工作总结
2015/04/22 职场文书
2015人事行政工作总结范文
2015/05/21 职场文书
Python OpenCV快速入门教程
2021/04/17 Python
Python基础之pandas数据合并
2021/04/27 Python