Yahoo!網站效能最佳體驗的34條黃金守則——JavaScript和CSS

bill.kang發表於2012-11-26

原文地址:http://dudo.org/archives/2008051417218.html

 

英文地址:http://developer.yahoo.com/performance/rules.html
中文地址:http://www.dudo.org/article.asp?id=216
      在第一部分和第二部分中我們分別介紹了改善網站效能中頁面內容和伺服器的幾條守則,除此之外,JavaScript和CSS也是我們頁面中經常用到的內容,對它們的優化也提高網站效能的重要方面:
CSS:

  1. 把樣式表置於頂部
  2. 避免使用CSS表示式(Expression)
  3. 使用外部JavaScript和CSS
  4. 削減JavaScript和CSS
  5. 用<link>代替@import
  6. 避免使用濾鏡

JavaScript

  1. 把指令碼置於頁面底部
  2. 使用外部JavaScript和CSS
  3. 削減JavaScript和CSS
  4. 剔除重複指令碼
  5. 減少DOM訪問
  6. 開發智慧事件處理程式

17、把樣式表置於頂部
      在研究Yahoo!的效能表現時,我們發現把樣式表放到文件的<head />內部似乎會加快頁面的下載速度。這是因為把樣式表放到<head />內會使頁面有步驟的載入顯示。
      注重效能的前端伺服器往往希望頁面有秩序地載入。同時,我們也希望瀏覽器把已經接收到內容儘可能顯示出來。這對於擁有較多內容的頁面和網速較慢的使用者來說特別重要。向使用者返回視覺化的反饋,比如程式指標,已經有了較好的研究並形成了正式文件。在我們的研究中HTML頁面就是程式指標。當瀏覽器有序地載入檔案頭、導航欄、頂部的logo等對於等待頁面載入的使用者來說都可以作為視覺化的反饋。這從整體上改善了使用者體驗。
      把樣式表放在文件底部的問題是在包括Internet Explorer在內的很多瀏覽器中這會中止內容的有序呈現。瀏覽器中止呈現是為了避免樣式改變引起的頁面元素重繪。使用者不得不面對一個空白頁面。
      HTML規範清楚指出樣式表要放包含在頁面的<head />區域內:“和<a />不同,<link />只能出現在文件的<head />區域內,儘管它可以多次使用它”。無論是引起白屏還是出現沒有樣式化的內容都不值得去嘗試。最好的方案就是按照HTML規範在文件<head />內載入你的樣式表。

18、避免使用CSS表示式(Expression) 
      CSS表示式是動態設定CSS屬性的強大(但危險)方法。Internet Explorer從第5個版本開始支援CSS表示式。下面的例子中,使用CSS表示式可以實現隔一個小時切換一次背景顏色:
      background-color: expression( (new Date()).getHours()%2 ? “#B8D4FF” : “#F08A00” ); 
如上所示,expression中使用了JavaScript表示式。CSS屬性根據JavaScript表示式的計算結果來設定。expression方法在其它瀏覽器中不起作用,因此在跨瀏覽器的設計中單獨針對Internet Explorer設定時會比較有用。
      表示式的問題就在於它的計算頻率要比我們想象的多。不僅僅是在頁面顯示和縮放時,就是在頁面滾動、乃至移動滑鼠時都會要重新計算一次。給CSS表示式增加一個計數器可以跟蹤表示式的計算頻率。在頁面中隨便移動滑鼠都可以輕鬆達到10000次以上的計算量。
      一個減少CSS表示式計算次數的方法就是使用一次性的表示式,它在第一次執行時將結果賦給指定的樣式屬性,並用這個屬性來代替CSS表示式。如果樣式屬性必須在頁面週期內動態地改變,使用事件控制程式碼來代替CSS表示式是一個可行辦法。如果必須使用CSS表示式,一定要記住它們要計算成千上萬次並且可能會對你頁面的效能產生影響。

19、使用外部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,但是在頁面下載完成後動態下載外部檔案,在子頁面中使用到這些檔案時,它們已經快取到瀏覽器了。

20、削減JavaScript和CSS 
      精簡是指從去除程式碼不必要的字元減少檔案大小從而節省下載時間。消減程式碼時,所有的註釋、不需要的空白字元(空格、換行、tab縮排)等都要去掉。在JavaScript中,由於需要下載的檔案體積變小了從而節省了響應時間。精簡JavaScript中目前用到的最廣泛的兩個工具是JSMinYUI Compressor。YUI Compressor還可用於精簡CSS。
      混淆是另外一種可用於原始碼優化的方法。這種方法要比精簡複雜一些並且在混淆的過程更易產生問題。在對美國前10大網站的調查中發現,精簡也可以縮小原來程式碼體積的21%,而混淆可以達到25%。儘管混淆法可以更好地縮減程式碼,但是對於JavaScript來說精簡的風險更小。
      除消減外部的指令碼和樣式表檔案外,<script>和<style>程式碼塊也可以並且應該進行消減。即使你用Gzip壓縮過指令碼和樣式表,精簡這些檔案仍然可以節省5%以上的空間。由於JavaScript和CSS的功能和體積的增加,消減程式碼將會獲得益處。

21、用<link>代替@import
      前面的最佳實現中提到CSS應該放置在頂端以利於有序載入呈現。
      在IE中,頁面底部@import和使用<link>作用是一樣的,因此最好不要使用它。

22、避免使用濾鏡
      IE獨有屬性AlphaImageLoader用於修正7.0以下版本中顯示PNG圖片的半透明效果。這個濾鏡的問題在於瀏覽器載入圖片時它會終止內容的呈現並且凍結瀏覽器。在每一個元素(不僅僅是圖片)它都會運算一次,增加了記憶體開支,因此它的問題是多方面的。
      完全避免使用AlphaImageLoader的最好方法就是使用PNG8格式來代替,這種格式能在IE中很好地工作。如果你確實需要使用AlphaImageLoader,請使用下劃線_filter又使之對IE7以上版本的使用者無效。

23、把指令碼置於頁面底部
      指令碼帶來的問題就是它阻止了頁面的平行下載。HTTP/1.1 規範建議,瀏覽器每個主機名的並行下載內容不超過兩個。如果你的圖片放在多個主機名上,你可以在每個並行下載中同時下載2個以上的檔案。但是當下載指令碼時,瀏覽器就不會同時下載其它檔案了,即便是主機名不相同。
      在某些情況下把指令碼移到頁面底部可能不太容易。比如說,如果指令碼中使用了document.write來插入頁面內容,它就不能被往下移動了。這裡可能還會有作用域的問題。很多情況下,都會遇到這方面的問題。
      一個經常用到的替代方法就是使用延遲指令碼。DEFER屬性表明指令碼中沒有包含document.write,它告訴瀏覽器繼續顯示。不幸的是,Firefox並不支援DEFER屬性。在Internet Explorer中,指令碼可能會被延遲但效果也不會像我們所期望的那樣。如果指令碼可以被延遲,那麼它就可以移到頁面的底部。這會讓你的頁面載入的快一點。

24、剔除重複指令碼
      在同一個頁面中重複引用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檔案頭等。

25、減少DOM訪問
      使用JavaScript訪問DOM元素比較慢,因此為了獲得更多的應該頁面,應該做到:

  • 快取已經訪問過的有關元素
  • 線下更新完節點之後再將它們新增到文件樹中
  • 避免使用JavaScript來修改頁面佈局

      有關此方面的更多資訊請檢視Julien Lecomte在YUI專題中的文章“高效能Ajax應該程式”

26、開發智慧事件處理程式
      有時候我們會感覺到頁面反應遲鈍,這是因為DOM樹元素中附加了過多的事件控制程式碼並且些事件句病被頻繁地觸發。這就是為什麼說使用event delegation(事件代理)是一種好方法了。如果你在一個div中有10個按鈕,你只需要在div上附加一次事件控制程式碼就可以了,而不用去為每一個按鈕增加一個控制程式碼。事件冒泡時你可以捕捉到事件並判斷出是哪個事件發出的。
      你同樣也不用為了操作DOM樹而等待onload事件的發生。你需要做的就是等待樹結構中你要訪問的元素出現。你也不用等待所有影像都載入完畢。
      你可能會希望用DOMContentLoaded事件來代替onload,但是在所有瀏覽器都支援它之前你可使用YUI 事件應用程式中的onAvailable方法。
      有關此方面的更多資訊請檢視Julien Lecomte在YUI專題中的文章“高效能Ajax應該程式”

adpics.aspx?source=kbh1983&sourcesuninfo

相關文章