摘要:簡化終止條件上述情況,我們也可以不使用減值迭代,即像上文提到過的,在初始化時即將迭代長度賦值給一個局部變量。優化操作操作是最拖累性能的一方面,優化操作可以顯著提高性能。
Javascript最初是解釋型語言,現在,主流瀏覽器內置的Javascript引擎基本上都實現了Javascript的編譯執行,即使如此,我們仍需要優化自己寫的Javascript代碼,以獲得最佳性能。
注意作用域 避免全局作用域在之前的文章Javascript 變量、作用域和內存問題提到過,由于訪問變量需要在作用域鏈上進行查找,相比于局部變量,訪問全局變量的開銷更大,因此以下代碼:
var person = { name: "Sue", hobbies: ["Yoga", "Jogging"] }; function hobby() { for(let i=0; i可以進行如下優化:
function hobby() { let hobbies = person.hobbies; for(let i=0; i把需要頻繁訪問的全局變量賦值到局部變量中,可以減小查找深度,進而優化性能。
避免使用with
當然,上述優化過的代碼仍然有不足的地方,后面的部分會提到。為什么避免使用with?
with并不是必須的,使用局部變量可以達到同樣的目的
with創建了自己的作用域,相當于增加了作用域內部查找變量的深度
舉一個例子:
function test() { var innerW = ""; var outerW = ""; with(window) { innerW = innerWidth; outerW = outerWidth; } return "Inner W: " + innerW + ", Outer W: " + outerW; } test() // "Inner W: 780, Outer W: 795"上述代碼中,with作用域減小了對全局變量window的查找深度,不過與此同時,也增加了作用域中局部變量innerW和outerW的查找深度,功過相抵。
因此我們不如使用局部變量替代with:function test() { var w = window; var innerW = w.innerWidth; var outerW = w.outerWidth; return "Inner W: " + innerW + ", Outer W: " + outerW; }上述代碼仍然不是最優的。
算法復雜度一下表格列出了幾種算法復雜度:
O(1)
復雜度 名稱 描述 O(1) 常數 無論多少值,執行時間恒定,比如使用簡單值或訪問存貯在變量中的值 O(lg n) 對數 總執行時間與值的數量相關,但不一定需要遍歷每一個值 O(n) 線性 總執行時間與值的數量線性相關 O(n2) 平方 總執行時間與值的數量相關,每個值要獲取n次 如果我們直接使用字面量,或者訪問保存在變量中的值,時間復雜度為O(1),比如:
var value = 5; var sum = 10 + value;上述代碼進行了三次常量查找,分別是5,10,value,這段代碼整體復雜度為O(1)
訪問數組也是時間復雜度為O(1)的操作,以下代碼整體復雜度為O(1):var values = [1, 2]; var sum = values[0] + values[1];避免不必要的屬性查找在對象上訪問屬性是一個O(n)的操作,Javascript 面向對象的程序設計(原型鏈與繼承)文中提到過,訪問對象中的屬性時,需要沿著原型鏈追溯查找,屬性查找越多,執行時間越長,比如:
var persons = ["Sue", "Jane", "Ben"]; for(let i=0; i上述代碼中,每次循環都會比較i
,為了避免頻繁的屬性查找,可以進行如下優化: var persons = ["Sue", "Jane", "Ben"]; for(let i=0, len = persons.length; i即如果循環長度在循環開始時即可確定,就將要循環的長度在初始化的時候聲明為一個局部變量。
優化循環由于循環時反復執行的代碼,動輒上百次,因此優化循環時性能優化中很重要的部分。
減值迭代
為什么要進行減值迭代,我們比較如下兩個循環:
var nums = [1, 2, 3, 4]; for(let i=0; i for(let i=nums.length-1; i>-1; i--) { console.log(nums[i]); }二者有如下區別:
迭代順序不同
前者支持動態增減數組元素,后者不支持
后者性能優于前者,前者每次循環都會計算nums.length,頻繁的屬性查找降低性能
因此,出于性能的考慮,如果不在乎順序,迭代長度初始即可確定,使用減值迭代更優。
簡化終止條件
上述情況,我們也可以不使用減值迭代,即像上文提到過的,在初始化時即將迭代長度賦值給一個局部變量。
簡化循環體
循環體應最大程度地被優化,避免進行不必要的密集的計算
使用while循環
為什么使用while循環,我們可以比較如下兩個循環:
var len = nums.length; for(let i=0; i整合冒泡事件處理 var i = nums.length ; while(--len > -1) { console.log(nums[len]); }以上兩個循環有一個很明顯的不同點:while循環將每次循環終止條件的判斷和index的自增合并為一個語句,在后續部分會講解語句數量與性能優化的關系。
展開循環由于建立循環和處理終止條件需要額外的開銷,因此如果循環次數比較少,而且可以確定,我們可以將其展開,比如:
process(nums[0]); process(nums[1]);如果迭代次數不能事先確定,可以使用Duff裝置,其中比較著名的是Andrew B. King提出的一種Duff技術,通過計算迭代次數是否為8的倍數將循環展開,將“零頭”與“整數”分成兩個多帶帶的do-while循環,在處理大數據集時優化效果顯著:
var iterations = Math.floor(values.length / 8); var leftover = values.length % 8; var i = 0; if (leftover > 0){ do { process(values[i++]); } while (--leftover > 0); } do { process(values[i++]); process(values[i++]); process(values[i++]); process(values[i++]); process(values[i++]); process(values[i++]); process(values[i++]); process(values[i++]); } while (--iterations > 0);避免雙重解釋eval() Function() setTimeout()可以傳入字符串,Javascript引擎會將其解析成可以執行的代碼,意味著,Javascript執行到這里需要額外開一個解釋器來解析字符串,會明顯降低性能,因此:
盡量避免使用eval()
避免使用Function構造函數,用一般function來代替
setTimeout()傳入函數作為參數
其他使用原生方法
原生方法都是用C/C++之類的編譯語言寫出來的,比Javascript快得多。
使用switch語句
多個if-else可以轉換為switch語句,還可以按照最可能到最不可能排序case
使用位運算符
當進行數學運算的時候,位運算操作要比任何布爾運算或者算數運算快。選擇性地用位運算替換算數運算可以極大提升復雜計算的性能。諸如取模,邏輯與和邏輯或都可
以考慮用位運算來替換。書中的這段話筆者表示不能理解,由于使用&& ||做邏輯判斷時,有的時候只需要求得第一個表達式的結果便可以結束運算,而& |無論如何都要求得兩個表達式的結果才可以結束運算,因此后者的性能沒有占太大優勢。
這里,補充一下位運算符如何發揮邏輯運算符的功能,首先看幾個例子:7 === 7 & 6 === 6 1 7 === 7 & 5 === 4 0 7 === 7 | 6 ===6 1 7 === 7 | 7 ===6 1 7 === 6 | 6 === 5 0也許你會恍然大悟,位運算符并沒有產生true 或 false,它只是利用了Number(true) === 1 Number(false) === 0 Boolean(1) === true Boolean(0) === false。
最小化語句數Javascript代碼中的語句數量會影響執行的速度,盡量組合語句,可以減少腳本的執行時間。
多個變量聲明當我們需要聲明多個變量,比如:
var name = ""; var age = 18; var hobbies = [];可以做如下優化:
var name = "", age = 18, hobbies = [];合并迭代值上文中我們提到一個例子,使用while循環可以合并自減和判斷終止條件,我們還可以換一種寫法:
var i = nums.length ; while(len > -1) { console.log(nums[len--]); }即將自減與使用index取值合并為一個語句。
使用字面量創建數組和對象即將如下代碼:
var array = new Array(); array[0] = 1; array[1] = 2; var person = new Object(); person.name = "Sue"; person.age = 18;替換成:
var array = [1, 2]; var person = { name:"Sue", age:18 };省了4行代碼。
優化DOM操作DOM操作是最拖累性能的一方面,優化DOM操作可以顯著提高性能。
最小化現場更新的次數如果我們要修改的DOM已經顯示在頁面,那么我們就是在做現場更新,由于每次更新瀏覽器都要重新計算,重新渲染,非常消耗性能,因此我們應該最小化現場更新的次數,比如我們要向頁面添加一個列表:
var body = document.getElementsByTagName("body")[0]; for(let i=0; i<10; i++) { item = document.createElement("span"); body.appendChild(item); item.appendChild(document.createTextNode("Item" + i)); }每次循環時都會進行兩次現場更新,添加div,為div添加文字,總共需要20次現場更新,頁面要重繪20次。
現場更新的性能瓶頸不在于更新的大小,而在于更新的次數,因此,我們可以將所有的更新一次繪制到頁面上,有以下兩個方法:文檔片段
可以使用文檔片段先收集好要添加的元素,最后在父節點上調用appendChild()將片段的子節點添加到父節點中,注意,片段本身不會被添加。
thisinnerHTML
使用innerHTML與使用諸如createElement() appendChild()方法有一個顯著的區別,前者使用內部的DOM來創建DOM結構,后者使用JavaScript的DOM來創建DOM結構,前者要快得多,之前的例子用innerHTML改寫為:
var ul = document.getElementById("ul"), innerHTML = ""; for(let i=0; i<10; i++) { innerHTML += "Item " + i + " "; } ul.innerHTML = innerHTML;頁面上的事件處理程序數量與頁面相應用戶交互的速度之間存在負相關,具體原因有多方面:
創建函數會占用內存
綁定事件處理方法時,需要訪問DOM
因此對于冒泡事件,盡可能由父元素甚至祖先元素代子元素處理,這樣一個事件處理方法可以負責多個目標的事件處理,比如:
注意HTMLCollectionthis訪問HTMLCollection的代價非常昂貴。
下面的每個項目(以及它們指定的屬性)都返回 HTMLCollection:Document (images, applets, links, forms, anchors)
form (elements)
map (areas)
select (options)
table (rows, tBodies)
tableSection (rows)
row (cells)
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/52305.html
摘要:本文是圖說系列文章的第五篇。這樣的話,使用的開發者也不需要做任何適配,但是它們卻能獲得更高性能。該圖并不是用來準確的衡量其性能的。運行編寫出高性能的代碼是可能的。這種清理工作由引擎自動進行,稱為垃圾回收。 本文是圖說 WebAssembly 系列文章的第五篇。如果您還未閱讀之前的文章,建議您從第一篇入手。 在上一篇文章中,我們說到了使用 WebAssembly 和 JavaScript...
摘要:下面我們撇開網絡方面的優化,只分析靜態資源方面的優化。不過,也會阻止的構建和延緩網頁渲染。未優化正常加載優化后異步加載根據上面的分析,我們可以清楚的認識到,非必要優先加載的,選擇異步加載是最優選擇。 為什么做優化 經典問題:白屏時間過長,用戶體驗差產生的原因:網絡問題、關鍵渲染路徑(CRP)問題 怎么做優化 如何做好優化呢,網上隨便一搜,就有很多優化總結,無非就是網絡優化、靜態資源(h...
摘要:下面我們撇開網絡方面的優化,只分析靜態資源方面的優化。不過,也會阻止的構建和延緩網頁渲染。未優化正常加載優化后異步加載根據上面的分析,我們可以清楚的認識到,非必要優先加載的,選擇異步加載是最優選擇。 為什么做優化 經典問題:白屏時間過長,用戶體驗差產生的原因:網絡問題、關鍵渲染路徑(CRP)問題 怎么做優化 如何做好優化呢,網上隨便一搜,就有很多優化總結,無非就是網絡優化、靜態資源(h...
摘要:下面我們撇開網絡方面的優化,只分析靜態資源方面的優化。不過,也會阻止的構建和延緩網頁渲染。未優化正常加載優化后異步加載根據上面的分析,我們可以清楚的認識到,非必要優先加載的,選擇異步加載是最優選擇。 為什么做優化 經典問題:白屏時間過長,用戶體驗差產生的原因:網絡問題、關鍵渲染路徑(CRP)問題 怎么做優化 如何做好優化呢,網上隨便一搜,就有很多優化總結,無非就是網絡優化、靜態資源(h...
摘要:插件性能優化及個人常用優化方法經常會觸發視覺變化。作用域鏈指的是當前作用于下可用變量的集合,它在各種主流瀏覽器中至少包含兩個部分局部變量的集合和全局變量的集合。在考慮優化時,數值和變量的性能差不多,并且速度顯著優于對象屬性和數組元素。 JavaScript 插件性能優化及個人react常用優化方法 JavaScript 經常會觸發視覺變化。有時是直接通過樣式操作,有時是會產生視覺變化...
閱讀 3626·2021-11-24 10:22
閱讀 3695·2021-11-22 09:34
閱讀 2498·2021-11-15 11:39
閱讀 1536·2021-10-14 09:42
閱讀 3669·2021-10-08 10:04
閱讀 1564·2019-08-30 15:52
閱讀 854·2019-08-30 13:49
閱讀 3025·2019-08-30 11:21