jQuery高階技巧——效能優化篇
作者:Lyzh
原文地址:http://blog.linjiajun.com/2015/08/23/jQuery高階技巧——效能優化篇/
通過CDN(Content Delivery Network)引入jQuery庫
要提升網站中javascript的效能,最簡單的一步就是引入最新版本的jQuery庫。新發布的版本,通常在效能上會有更好的提升,而且也修復了一下bug。或者通過CDN引入,也是很好的選擇。通過CDN引入能夠減少網站的載入時間。以下是一些CDN服務:
<!– 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服務:
<!–新浪 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樣式屬性也是同樣的道理。
更多閱讀: 前端頁面卡頓?可能是DOM操作惹的禍,你需要優化程式碼
適當使用原生JS
建立jQuery物件會帶來一些開銷。所以,如果比較注重效能的話,儘可能使用原生的javascript。在某些方面可能會更容易理解和寫更少的程式碼。
例如:
// 列印list中的li的id
$(‘#colors li’ ). each(function(){
//將$(this).attr(‘id’)方法替換為直接通過ID屬性訪問
console. log(this. id);
})
選擇器優化
如果你需要更好的效能,但是仍然要用到jQuery,你可以在jQuery選擇器優化做一些嘗試。以下是一個測試程式,通過瀏覽器的控制檯console.time
和console.timeEnd
方法來記錄不同選擇器執行時間。
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’);
結果:
可以看到通過陣列實現方式遍歷,執行效率更高。
以上是一些蒐集知識的總結,如有任何建議或疑問,歡迎留言討論。
相關文章
- spark效能調優指南高階篇Spark
- MySQL 優化一(高階篇)MySql優化
- MySQL 優化(二)(高階篇)MySql優化
- MySQL 優化三(優化規則)(高階篇)MySql優化
- [java][效能優化]java高階開發必會的50個效能優化Java優化
- SQL效能優化技巧SQL優化
- 效能優化篇優化
- Hive高階優化Hive優化
- MySQL 高階優化MySql優化
- IOS效能優化篇iOS優化
- Android效能優化(1)—webview優化篇Android優化WebView
- mpvue效能優化實戰技巧Vue優化
- web效能常見優化技巧Web優化
- Tomcat高階特性及效能調優Tomcat
- iOS 效能篇一一UITableView效能優化iOSUIView優化
- Mysql高階優化(一)MySql優化
- Mysql高階優化(二)MySql優化
- 前端效能優化JavaScript篇前端優化JavaScript
- Android效能優化篇之服務優化Android優化
- CSS效能優化的8個技巧CSS優化
- CSS效能優化的幾個技巧CSS優化
- Linux 效能優化之 CPU 篇 ----- 套路篇Linux優化
- 效能優化篇 - Performance(工具 & api)優化ORMAPI
- UITableView效能優化-中級篇UIView優化
- Linux 效能優化之 cup 篇Linux優化
- Linux 效能優化之 IO 篇Linux優化
- 效能優化開篇綜述優化
- PHP效能優化 -理論篇PHP優化
- Java高階開發必會的50個效能優化細節Java優化
- Redis基礎、高階特性與效能調優Redis
- Redis 基礎、高階特性與效能調優Redis
- JavaScript 高階技巧JavaScript
- Nginx 高階篇(七)大訪問量優化的整體思路Nginx優化
- [譯] 強化學習入門篇:Simmer 模擬平臺高階使用技巧強化學習
- 前端效能優化之路——圖片篇。前端優化
- iOS效能優化系列篇之“列表流暢度優化”iOS優化
- iOS效能優化系列篇之“優化總體原則”iOS優化
- 前端進階(1)Web前端效能優化前端Web優化
- Android高階效能調優;不可思議的OOM!AndroidOOM