Yahoo!的Exceptional Performance團隊爲改善Web性能帶來最佳實踐。他們爲此進行了 一系列的實驗、開發了各類工具、寫了大量的文章和博客並在各類會議上參與探討。最佳實踐的核心就是旨在提升網站性能。javascript
Excetional Performance 團隊總結出了一系列能夠提升網站速度的方法。能夠分爲 7 大類 34 條。包括內容、服務器、cookie、CSS、JavaScript、圖片、移動應用等七部分。php
本文爲CSS和Javascript部分:前端
除此以外,JavaScript 和 CSS 也是咱們頁面中常常用到的內容,對它們的優化也提升網 站性能的重要方面:java
CSS:express
1. 把樣式表置於頂部瀏覽器
2. 避免使用CSS表達式(Expression)緩存
3. 使用外部JavaScript和CSS服務器
4. 削減JavaScript和CSScookie
5. 用<link>代替@importide
6. 避免使用濾鏡
JavaScript
7. 把腳本置於頁面底部
8. 使用外部JavaScript和CSS
削減JavaScript和CSS 和 剔除重複腳本(參考三、4)
9. 減小DOM訪問
10. 開發智能事件處理程序
一、把樣式表置於頂部
在研究Yahoo!的性能表現時,咱們發現把樣式表放到文檔的<head />內部彷佛會加快頁 面的下載速度。這是由於把樣式表放到<head />內會使頁面有步驟的加載顯示。
注重性能的前端服務器每每但願頁面有秩序地加載。同時,咱們也但願瀏覽器把已經接收到內容儘量顯示出來。這對於擁有較多內容的頁面和網速較慢的用戶來講特別重要。
向用戶返回可視化的反饋,好比進程指針,已經有了較好的研究並造成了正式文檔。在咱們的研究中HTML頁面就是進程指針。當瀏覽器有序地加載文件頭、導航欄、頂部的logo等對於等待頁面加載的用戶來講均可以做爲可視化的反饋。這從總體上改善了用戶體驗。
把樣式表放在文檔底部的問題是在包括Internet Explorer在內的不少瀏覽器中這會中 止內容的有序呈現。瀏覽器停止呈現是爲了不樣式改變引發的頁面元素重繪。用戶不 得不面對一個空白頁面。
HTML規範清楚指出樣式表要放包含在頁面的<head />區域內:"和<a />不一樣,<link /> 只能出如今文檔的<head />區域內,儘管它能夠屢次使用它"。不管是引發白屏仍是出 現沒有樣式化的內容都不值得去嘗試。最好的方案就是按照HTML規範在文檔<head />內加載你的樣式表。
二、避免使用CSS表達式(Expression)
CSS表達式是動態設置CSS屬性的強大(但危險)方法。Internet Explorer從第5個版 本開始支持CSS表達式。下面的例子中,使用CSS表達式能夠實現隔一個小時切換一次背景顏色:
如上所示,expression中使用了JavaScript表達式。CSS屬性根據JavaScript表達式的 計算結果來設置。expression方法在其它瀏覽器中不起做用,所以在跨瀏覽器的設計中
單獨針對Internet Explorer設置時會比較有用。 表達式的問題就在於它的計算頻率要比咱們想象的多。不只僅是在頁面顯示和縮放時,就是在頁面滾動、乃至移動鼠標時都會要從新計算一次。給CSS表達式增長一個計數器能夠跟蹤表達式的計算頻率。在頁面中隨便移動鼠標均可以輕鬆達到 10000 次以上的計算量。
一個減小CSS表達式計算次數的方法就是使用一次性的表達式,它在第一次運行時將結 果賦給指定的樣式屬性,並用這個屬性來代替CSS表達式。若是樣式屬性必須在頁面周 期內動態地改變,使用事件句柄來代替CSS表達式是一個可行辦法。若是必須使用CSS表達式,必定要記住它們要計算成千上萬次而且可能會對你頁面的性能產生影響。
三、使用外部JavaScript和CSS
不少性能規則都是關於如何處理外部文件的。可是,在你採起這些措施前你可能會問到 一個更基本的問題:JavaScript和CSS是應該放在外部文件中呢仍是把它們放在頁面本 身以內呢?
在實際應用中使用外部文件能夠提升頁面速度,由於JavaScript和CSS文件都能在瀏覽 器中產生緩存。內置在HTML文檔中的JavaScript和CSS則會在每次請求中隨HTML文檔重 新下載。這雖然減小了HTTP請求的次數,卻增長了HTML文檔的大小。從另外一方面來講, 若是外部文件中的JavaScript和CSS被瀏覽器緩存,在沒有增長HTTP請求次數的同時可 以減小HTML文檔的大小。
關鍵問題是,外部JavaScript和CSS文件緩存的頻率和請求HTML文檔的次數有關。雖然 有必定的難度,可是仍然有一些指標能夠一測量它。若是一個會話中用戶會瀏覽你網站 中的多個頁面,而且這些頁面中會重複使用相同的腳本和樣式表,緩存外部文件就會帶 來更大的益處。
許多網站沒有功能創建這些指標。對於這些網站來講,最好的堅定方法就是把 JavaScript和CSS做爲外部文件引用。比較適合使用內置代碼的例外就是網站的主頁, 如Yahoo!主頁和My Yahoo!。主頁在一次會話中擁有較少(可能只有一次)的瀏覽量,你能夠發現內置JavaScript和CSS對於終端用戶來講會加快響應時間。
對於擁有較大瀏覽量的首頁來講,有一種技術能夠平衡內置代碼帶來的HTTP請求減小與 經過使用外部文件進行緩存帶來的好處。其中一個就是在首頁中內置JavaScript和CSS, 可是在頁面下載完成後動態下載外部文件,在子頁面中使用到這些文件時,它們已經緩 存到瀏覽器了。
四、削減JavaScript和CSS
精簡是指從去除代碼沒必要要的字符減小文件大小從而節省下載時間。消減代碼時,全部 的註釋、不須要的空白字符(空格、換行、tab縮進)等都要去掉。在JavaScript中, 因爲須要下載的文件體積變小了從而節省了響應時間。精簡JavaScript中目前用到的最 普遍的兩個工具是JSMin和YUI Compressor。YUI Compressor還可用於精簡CSS。
混淆是另一種可用於源代碼優化的方法。這種方法要比精簡複雜一些而且在混淆的過 程更易產生問題。在對美國前 10 大網站的調查中發現,精簡也能夠縮小原來代碼體積 的 21%,而混淆能夠達到 25%。儘管混淆法能夠更好地縮減代碼,可是對於JavaScript 來講精簡的風險更小。
除消減外部的腳本和樣式表文件外,<script>和<style>代碼塊也能夠而且應該進行消 減。即便你用Gzip壓縮過腳本和樣式表,精簡這些文件仍然能夠節省 5%以上的空間。因爲JavaScript和CSS的功能和體積的增長,消減代碼將會得到益處。
五、用<link>代替@import
前面的最佳實現中提到 CSS 應該放置在頂端以利於有序加載呈現。
在 IE 中,頁面底部@import 和使用<link>做用是同樣的,所以最好不要使用它。
六、避免使用濾鏡
IE 獨有屬性 AlphaImageLoader 用於修正 7.0 如下版本中顯示 PNG 圖片的半透明效果。 這個濾鏡的問題在於瀏覽器加載圖片時它會終止內容的呈現而且凍結瀏覽器。在每個 元素(不只僅是圖片)它都會運算一次,增長了內存開支,所以它的問題是多方面的。 徹底避免使用 AlphaImageLoader 的最好方法就是使用 PNG8 格式來代替,這種格式能在 IE 中很好地工做。若是你確實須要使用 AlphaImageLoader,請使用下劃線_filter 又使之對 IE7 以上版本的用戶無效。
七、把腳本置於頁面底部
腳本帶來的問題就是它阻止了頁面的平行下載。HTTP/1.1 規範建議,瀏覽器每一個主機 名的並行下載內容不超過兩個。若是你的圖片放在多個主機名上,你能夠在每一個並行下 載中同時下載 2 個以上的文件。可是當下載腳本時,瀏覽器就不會同時下載其它文件了, 即使是主機名不相同。
在某些狀況下把腳本移到頁面底部可能不太容易。好比說,若是腳本中使用了 document.write來插入頁面內容,它就不能被往下移動了。這裏可能還會有做用域的問 題。不少狀況下,都會遇到這方面的問題。 一個常常用到的替代方法就是使用延遲腳本。DEFER屬性代表腳本中沒有包含 document.write,它告訴瀏覽器繼續顯示。不幸的是,Firefox並不支持DEFER屬性。在 Internet Explorer中,腳本可能會被延遲但效果也不會像咱們所指望的那樣。若是腳
本能夠被延遲,那麼它就能夠移到頁面的底部。這會讓你的頁面加載的快一點。
八、剔除重複腳本
在同一個頁面中重複引用 JavaScript 文件會影響頁面的性能。你可能會認爲這種狀況
並很少見。對於美國前 10 大網站的調查顯示其中有兩家存在重複引用腳本的狀況。有 兩種主要因素致使一個腳本被重複引用的奇怪現象發生:團隊規模和腳本數量。若是真 的存在這種狀況,重複腳本會引發沒必要要的 HTTP 請求和無用的 JavaScript 運算,這降 低了網站性能。
在 Internet Explorer 中會產生沒必要要的 HTTP 請求,而在 Firefox 卻不會。在 Internet Explorer 中,若是一個腳本被引用兩次並且它又不可緩存,它就會在頁面加載過程當中產 生兩次 HTTP 請求。即時腳本能夠緩存,當用戶重載頁面時也會產生額外的 HTTP 請求。
除增長額外的 HTTP 請求外,屢次運算腳本也會浪費時間。在 Internet Explorer 和 Firefox 中無論腳本是否可緩存,它們都存在重複運算 JavaScript 的問題。 一個避免偶爾發生的兩次引用同一腳本的方法是在模板中使用腳本管理模塊引用腳本。 在 HTML 頁面中使用<script />標籤引用腳本的最多見方法就是:
<script type="text/javascript" src="menu_1.0.17.js"></script>
在 PHP 中能夠經過建立名爲 insertScript 的方法來替代:
<?php insertScript("menu.js") ?>
爲了防止屢次重複引用腳本,這個方法中還應該使用其它機制來處理腳本,如檢查所屬 目錄和爲腳本文件名中增長版本號以用於 Expire 文件頭等。
九、減小 DOM 訪問
使用 JavaScript 訪問 DOM 元素比較慢,所以爲了得到更多的應該頁面,應該作到:
• 緩存已經訪問過的有關元素
• 線下更新完節點以後再將它們添加到文檔樹中
• 避免使用 JavaScript 來修改頁面佈局
有關此方面的更多信息請查看Julien Lecomte在YUI專題中的文章"高性能Ajax應該程序"。
十、開發智能事件處理程序
有時候咱們會感受到頁面反應遲鈍,這是由於DOM樹元素中附加了過多的事件句柄而且 些事件句病被頻繁地觸發。這就是爲何說使用event delegation(事件代理)是一種 好方法了。若是你在一個div中有 10 個按鈕,你只須要在div上附加一次事件句柄就可 以了,而不用去爲每個按鈕增長一個句柄。事件冒泡時你能夠捕捉到事件並判斷出是 哪一個事件發出的。
你一樣也不用爲了操做DOM樹而等待onload事件的發生。你須要作的就是等待樹結構中 你要訪問的元素出現。你也不用等待全部圖像都加載完畢。 你可能會但願用DOMContentLoaded事件來代替onload,可是在全部瀏覽器都支持它以前 你可以使用YUI 事件應用程序中的onAvailable方法。
有關此方面的更多信息請查看Julien Lecomte在YUI專題中的文章"高性能Ajax應該程序"。