jQuery高級技巧——性能優化篇

 

經過CDN(Content Delivery Network)引入jQuery庫

 要提高網站中javascript的性能的最簡單的一步就是引入最新版本的jQuery庫,新發布的版本一般在性能上會有更好的提高並且也修復了一下bug。或者經過CDN引入也是很好的選擇,經過CDN引入可以減小網站的加載時間。如下是一些CDN服務:javascript

<!-- Case 1 - jQuery CDN -->
<script src="http://code.jquery.com/jquery-1.10.2.min.js" ></script>
<!-- Case 2 - requesting jQuery from Googles CDN (notice the protocol) -->
<script src="//ajax.googleapis.com/ajax/libs/jquery/1.10.2/jquery.min.js" ></script>
<!-- Case 3 - requesting the latest minor 1.10.x version (only cached for an hour) -->
<script src="//ajax.googleapis.com/ajax/libs/jquery/1.10/jquery.min.js" ></script>
<!-- Case 4 - requesting the absolute latest jQuery version (use with caution) -->
<script src="http://code.jquery.com/jquery.min.js" ></script>

一些國內的CDN服務:html

http://www.bootcdn.cn/jquery/

<!--新浪 CDN-->
<script src="http://lib.sinaapp.com/js/jquery/1.9.1/jquery-1.9.1.min.js"></script>

<!--百度 CDN-->
<script src="http://libs.baidu.com/jquery/1.9.1/jquery.min.js"></script>

<!--Bootstrap CDN-->
http://www.bootcdn.cn/jquery/

減小DOM操做

雖然javascript性能上有了很大的提高,可是DOM操做仍是很耗費資源的,須要減小對DOM操做。當在一個頁面中插入大量的元素的時候,尤爲重要。例如:前端

<div id="elem" ></div>
// 很差的方式
//var elem = $('#elem');
//for(var i = 0; i < 100; i++){
// elem.append('<li>element '+i+'</li>');
//}

// 好的方式
var elem = $('#elem' ),
arr = [];
for(var i = 0; i < 100; i++){
arr. push('<li>element ' +i+'</li>' );
}
elem. append(arr. join('' ));

將全部的元素緩存起來一次插入性能上會有所提高,由於只觸發頁面一次重繪。對於CSS樣式屬性也是一樣的道理。java

更多閱讀: 前端頁面卡頓?多是DOM操做惹的禍,你須要優化代碼jquery

適當使用原生JS

建立jQuery對象會帶來一些開銷。因此,若是比較注重性能的話,儘量使用原生的javascript。在某些方面可能會更容易理解和寫更少的代碼。例如:ajax

// 打印list中的li的id
$('#colors li' ). each(function(){
//將$(this).attr('id')方法替換爲直接經過ID屬性訪問
console. log(this. id);
})

選擇器優化

若是你須要更好的性能,可是仍然要用到jQuery,你能夠在jQuery選擇器優化作一些嘗試。如下是一個測試程序,經過瀏覽器的控制檯console.time 和console.timeEnd 方法來記錄不一樣選擇器執行時間。api

 HTML:數組

<div id="peanutButter" >
<div id="jelly" class=".jellyTime" ></div>
</div>

JS:瀏覽器

//測試程序
var iterations = 10000,
    i;

//--------------------------------------------
//Case 1: 很慢
console.time('Fancy');
for (i = 0; i < iterations; i++) {
    
    $('#peanutButter div:first');
}
console.timeEnd('Fancy');

//--------------------------------------------
//Case 2: 比較好,但仍然很慢
console.time('Parent-child');
for (i = 0; i < iterations; i++) {
$('#peanutButter div'); } console.timeEnd('Parent-child'); //-------------------------------------------- //Case 3: 一些瀏覽器會比較快 console.time('Parent-child by class'); for (i = 0; i < iterations; i++) { // 經過後代Class選擇器 $('#peanutButter .jellyTime'); } console.timeEnd('Parent-child by class'); //-------------------------------------------- //Case 4: 更好的方式 console.time('By class name'); 21 for (i = 0; i < iterations; i++) { // 直接經過Class選擇器 $('.jellyTime'); } console.timeEnd('By class name'); //-------------------------------------------- //Case 5: 推薦的方式 ID選擇器 console.time('By id'); for (i = 0; i < iterations; i++) { $('#jelly'); } console.timeEnd('By id');

執行結果:緩存

 

緩存jQuery對象

每次經過選擇器構建一個新的jQuery對象時,jQuery的核心部分的Sizzle引擎會遍歷DOM而後經過對應的選擇器來匹配真正的dom元素。這種方式比較低效,在現代瀏覽器中能夠經過document.querySelector方法經過傳入對應的Class參數來匹配對應的元素,不過IE8如下版本不支持此方法。一個提升性能的實踐是經過變量緩存jQuery對象。例如:

<ul id="pancakes" >
         <li>first</li>
         <li>second</li>
         <li>third</li>
         <li>fourth</li>
         <li>fifth</li>
</ul>

JS:

// 很差的方式:
// $('#pancakes li').eq(0).remove();
// $('#pancakes li').eq(1).remove();
// $('#pancakes li').eq(2).remove();
// ------------------------------------
// 推薦的方式:
var pancakes = $('#pancakes li');
pancakes.eq(0).remove();
pancakes.eq(1).remove();
pancakes.eq(2).remove();
// ------------------------------------
// 或者:
// pancakes.eq(0).remove().end()
//  .eq(1).remove().end()
//  .eq(2).remove().end();

定義一個能夠複用的函數

直接上例子:

HTML:

<button id="menuButton" >Show Menu!</button>
<a href="#" id="menuLink" >Show Menu!</a>

JS:

//Bad: 
//這個會致使多個回調函數的副本佔用內存
$('#menuButton, #menuLink' ). click(function(){
// ...
});

//----------------------------------------------
//Better
function showMenu(){
alert('Showing menu!' );
// Doing something complex here
}

$('#menuButton' ). click(showMenu);
$('#menuLink' ). click(showMenu);

若是定義一個內聯(inline)回調函數同時這個包含多個元素的jQuery對象(正如上面所說的第一個例子),對於這個集合中的每一個元素都會在內存中保存一個回調函數的副本。

用數組方式來遍歷jQuery 對象集合

你或許沒有注意到,可是在性能方面,對於jQuery each方法這種優雅實現是有代價的。有一個辦法可以更快地遍歷一個jQuery對象。就是經過數組來實現,jQuery對象集合就是一個類數組,具備length和value屬性。能夠經過程序來測試一下性能:

HTML:

<ul id="testList" >
   <li>Item</li>
   <li>Item</li>
   <li>Item</li>
   <li>Item</li> 
   <li>Item</li>
   <li>Item</li>
   <li>Item</li>
   <li>Item</li>
   <li>Item</li>
   <!-- add 50 more -->
</ul>

JS:

var arr = $('li'),
    iterations = 100000;
//------------------------------
// Array實現:    
console.time('Native Loop');
for (var z = 0; z < iterations; z++) {
    var length = arr.length;
    for (var i = 0; i < length; i++) {
        arr[i];
    }
}
console.timeEnd('Native Loop');

//------------------------------
// each實現:    
console.time('jQuery Each');
for (z = 0; z < iterations; z++) {
    arr.each(function(i, val) {
        this;
    });
}
console.timeEnd('jQuery Each');

結果:

能夠看到經過數組實現方式遍歷,執行效率更高。

//-------------------------------------------------------持續更新...

以上是一些蒐集知識的總結,若有任何建議或疑問,歡迎留言討論。

相關文章
相關標籤/搜索