1. Javascript
DOM
DOM操作應(yīng)該是腳本中最耗性能的一類操作晌砾,例如增加、修改哼勇、刪除 DOM元素或者對 DOM集合進行操作积担。如果腳本中包含了大量的 DOM操作則需要注意以下幾點:
- HTML Collection(HTML收集器,返回的是一個數(shù)組內(nèi)容信息)
在腳本中 document.images先誉、document.forms 褐耳、getElementsByTagName()返回的都是 HTMLCollection類型的集合渴庆,在平時使用的時候大多將它作為數(shù)組來使用把曼,因為它有 length屬性漓穿,也可以使用索引訪問每一個元素晃危。不過在訪問性能上則比數(shù)組要差很多,原因是這個集合并不是一個靜態(tài)的結(jié)果震叮,它表示的僅僅是一個特定的查詢苇瓣,每次訪問該集合時都會重新執(zhí)行這個查詢從而更新查詢結(jié)果偿乖。所謂的 “訪問集合” 包括讀取集合的 length屬性贪薪、訪問集合中的元素。
因此竣稽,當你需要遍歷 HTML Collection的時候毫别,盡量將它轉(zhuǎn)為數(shù)組后再訪問,以提高性能忘闻。即使不轉(zhuǎn)換為數(shù)組齐佳,也請盡可能少的訪問它债沮,例如在遍歷的時候可以將 length屬性疫衩、成員保存到局部變量后再使用局部變量。
- Reflow & Repaint
除了上面一點之外童芹, DOM操作還需要考慮瀏覽器的 Reflow和Repaint 假褪,因為這些都是需要消耗資源的近顷,具體的可以參加以下文章:
慎用 with
with(obj){ p = 1}; 代碼塊的行為實際上是修改了代碼塊中的執(zhí)行環(huán)境窒升,將obj放在了其作用域鏈的最前端饱须,在 with代碼塊中訪問非局部變量是都是先從 obj上開始查找,如果沒有再依次按作用域鏈向上查找譬挚,因此使用 with相當于增加了作用域鏈長度殴瘦。而每次查找作用域鏈都是要消耗時間的号杠,過長的作用域鏈會導(dǎo)致查找性能下降。
因此立帖,除非你能肯定在 with代碼中只訪問 obj中的屬性晓勇,否則慎用 with灌旧,替代的可以使用局部變量緩存需要訪問的屬性枢泰。
避免使用 eval和 Function
每次 eval 或 Function 構(gòu)造函數(shù)作用于字符串表示的源代碼時,腳本引擎都需要將源代碼轉(zhuǎn)換成可執(zhí)行代碼窿克。這是很消耗資源的操作 —— 通常比簡單的函數(shù)調(diào)用慢 100倍以上年叮。
eval 函數(shù)效率特別低玻募,由于事先無法知曉傳給 eval 的字符串中的內(nèi)容补箍,eval在其上下文中解釋要處理的代碼坑雅,也就是說編譯器無法優(yōu)化上下文衬横,因此只能有瀏覽器在運行時解釋代碼蜂林。這對性能影響很大。
Function 構(gòu)造函數(shù)比 eval略好矮锈,因為使用此代碼不會影響周圍代碼 ;但其速度仍很慢苞笨。
此外瀑凝,使用 eval和 Function也不利于Javascript 壓縮工具執(zhí)行壓縮。
減少作用域鏈查找(這方面設(shè)計到一些內(nèi)容的相關(guān)問題)
前文談到了作用域鏈查找問題谚中,這一點在循環(huán)中是尤其需要注意的問題宪塔。如果在循環(huán)中需要訪問非本作用域下的變量時請在遍歷之前用局部變量緩存該變量囊拜,并在遍歷結(jié)束后再重寫那個變量蝌麸,這一點對全局變量尤其重要,因為全局變量處于作用域鏈的最頂端艾疟,訪問時的查找次數(shù)是最多的来吩。
低效率的寫法:
// 全局變量
var globalVar = 1;
function myCallback(info){
for( var i = 100000; i--;){
//每次訪問 globalVar 都需要查找到作用域鏈最頂端,本例中需要訪問 100000 次
globalVar += i;
}
}
更高效的寫法:
// 全局變量
var globalVar = 1;
function myCallback(info){
//局部變量緩存全局變量
var localVar = globalVar;
for( var i = 100000; i--;){
//訪問局部變量是最快的
localVar += i;
}
//本例中只需要訪問 2次全局變量 蔽莱,在函數(shù)中只需要將 globalVar中內(nèi)容的值賦給localVar 中區(qū)
globalVar = localVar;
}
此外弟疆,要減少作用域鏈查找還應(yīng)該減少閉包的使用。
數(shù)據(jù)訪問
Javascript中的數(shù)據(jù)訪問包括直接量 (字符串盗冷、正則表達式 )怠苔、變量仪糖、對象屬性以及數(shù)組柑司,其中對直接量和局部變量的訪問是最快的,對對象屬性以及數(shù)組的訪問需要更大的開銷锅劝。當出現(xiàn)以下情況時攒驰,建議將數(shù)據(jù)放入局部變量:
- 對任何對象屬性的訪問超過 1次
- 對任何數(shù)組成員的訪問次數(shù)超過 1次
另外,還應(yīng)當盡可能的減少對對象以及數(shù)組深度查找故爵。
字符串拼接
在 Javascript中使用"+" 號來拼接字符串效率是比較低的玻粪,因為每次運行都會開辟新的內(nèi)存并生成新的字符串變量,然后將拼接結(jié)果賦值給新變量诬垂。與之相比更為高效的做法是使用數(shù)組的 join方法劲室,即將需要拼接的字符串放在數(shù)組中最后調(diào)用其 join方法得到結(jié)果。不過由于使用數(shù)組也有一定的開銷结窘,因此當需要拼接的字符串較多的時候可以考慮用此方法很洋。
關(guān)于 Javascript優(yōu)化的更詳細介紹請參考:
2. CSS選擇符
在大多數(shù)人的觀念中,都覺得瀏覽器對 CSS選擇符的解析式從左往右進行的隧枫,例如
#toc A { color: #444; }
這樣一個選擇符喉磁,如果是從右往左解析則效率會很高棺克,因為第一個 ID選擇基本上就把查找的范圍限定了,但實際上瀏覽器對選擇符的解析是從右往左進行的线定。如上面的選擇符娜谊,瀏覽器必須遍歷查找每一個 A標簽的祖先節(jié)點,效率并不像之前想象的那樣高斤讥。
3. HTML
對 HTML本身的優(yōu)化現(xiàn)如今也越來越多的受人關(guān)注了
4. Image壓縮
圖片壓縮是個技術(shù)活纱皆,不過現(xiàn)如今這方面的工具也非常多,壓縮之后往往能帶來不錯的效果芭商,具體的壓縮原理以及方法在《 Even Faster Web Sites》第10 章有很詳細的介紹派草,有興趣的可以去看看。