如何提升頁面加載速度(二)

續上一篇javascript

8、使用外部的JavaScript和CSS

內聯腳本或者樣式能夠減小HTTP請求,按理來講能夠提升頁面加載的速度。然而在實際狀況中,當腳本或者樣式是從外部引入的文件,瀏覽器就有可能緩存它們,從而在之後加載的時候可以直接使用緩存,而HTML文檔的大小減少,從而提升加載速度。css

影響因素:html

一、每一個用戶產生的頁面瀏覽量越少,內聯腳本和樣式的論據越強勢。譬如一個用戶每月只訪問你的網站一兩次,那麼這種狀況下內聯將會更好。而若是該用戶可以產生不少頁面瀏覽量,那麼緩存的樣式和腳本將會極大減小下載的時間,提交頁面加載速度。前端

二、若是你的網站不一樣的頁面之間使用的組件大體相同,那麼使用外部文件能夠提升這些組件的重用率。java

加載後下載node

有時候咱們但願內聯樣式和腳本,但又能夠爲接下來的頁面提供外部文件。那麼咱們能夠在頁面加載完成止嘔動態加載外部組件,以便用戶接下來的訪問。python

 function doOnload() {
      setTimeout("downloadFile()",1000);
  }

  window.onload = doOnload;

  function downloadFile() {
      downloadCss("http://abc.com/css/a.css");
      downloadJS("http://abc.com/js/a.js");
 }

 function downloadCss(url) {
     var ele = document.createElement('link');
     ele.rel = "stylesheet";
     ele.type = "text/css";
     ele.href = url;

     document.body.appendChild(ele);
 }

 function downloadJS(url) {
     var ele = document.createElement('script');
     ele.src = url;
     document.body.appendChild(ele);
 }

在該頁面中,JavaScript和CSS被加載兩次(內聯和外部)。要使其正常工做,必須處理雙重定義。將這些組件放到一個不可見的IFrame中是一個比較好的解決方式。jquery

9、減小DNS查找

當咱們在瀏覽器的地址欄輸入網址(譬如: www.linux178.com) ,而後回車,回車這一瞬間到看到頁面到底發生了什麼呢?linux

域名解析 --> 發起TCP的3次握手 --> 創建TCP鏈接後發起http請求 --> 服務器響應http請求,瀏覽器獲得html代碼 --> 瀏覽器解析html代碼,並請求html代碼中的資源(如js、css、圖片等) --> 瀏覽器對頁面進行渲染呈現給用戶web

域名解析是頁面加載的第一步,那麼域名是如何解析的呢?以Chrome爲例:

1.Chrome瀏覽器 會首先搜索瀏覽器自身的DNS緩存(緩存時間比較短,大概只有1分鐘,且只能容納1000條緩存),看自身的緩存中是否有www.linux178.com 對應的條目,並且沒有過時,若是有且沒有過時則解析到此結束。
注:咱們怎麼查看Chrome自身的緩存?可使用 chrome://net-internals/#dns 來進行查看

2.若是瀏覽器自身的緩存裏面沒有找到對應的條目,那麼Chrome會搜索操做系統自身的DNS緩存,若是找到且沒有過時則中止搜索解析到此結束.
注:怎麼查看操做系統自身的DNS緩存,以Windows系統爲例,能夠在命令行下使用 ipconfig /displaydns 來進行查看

3.若是在Windows系統的DNS緩存也沒有找到,那麼嘗試讀取hosts文件(位於C:WindowsSystem32driversetc),看看這裏面有沒有該域名對應的IP地址,若是有則解析成功。

4.若是在hosts文件中也沒有找到對應的條目,瀏覽器就會發起一個DNS的系統調用,就會向本地配置的首選DNS服務器(通常是電信運營商提供的,也可使用像Google提供的DNS服務器)發起域名解析請求(經過的是UDP協議向DNS的53端口發起請求,這個請求是遞歸的請求,也就是運營商的DNS服務器必須得提供給咱們該域名的IP地址),運營商的DNS服務器首先查找自身的緩存,找到對應的條目,且沒有過時,則解析成功。若是沒有找到對應的條目,則有運營商的DNS代咱們的瀏覽器發起迭代DNS解析請求,它首先是會找根域的DNS的IP地址(這個DNS服務器都內置13臺根域的DNS的IP地址),找打根域的DNS地址,就會向其發起請求(請問www.linux178.com這個域名的IP地址是多少啊?),根域發現這是一個頂級域com域的一個域名,因而就告訴運營商的DNS我不知道這個域名的IP地址,可是我知道com域的IP地址,你去找它去,因而運營商的DNS就獲得了com域的IP地址,又向com域的IP地址發起了請求(請問www.linux178.com這個域名的IP地址是多少?),com域這臺服務器告訴運營商的DNS我不知道www.linux178.com這個域名的IP地址,可是我知道linux178.com這個域的DNS地址,你去找它去,因而運營商的DNS又向linux178.com這個域名的DNS地址(這個通常就是由域名註冊商提供的,像萬網,新網等)發起請求(請問www.linux178.com這個域名的IP地址是多少?),這個時候linux178.com域的DNS服務器一查,誒,果然在我這裏,因而就把找到的結果發送給運營商的DNS服務器,這個時候運營商的DNS服務器就拿到了www.linux178.com這個域名對應的IP地址,並返回給Windows系統內核,內核又把結果返回給瀏覽器,終於瀏覽器拿到了www.linux178.com對應的IP地址,該進行一步的動做了。

注:通常狀況下是不會進行如下步驟的

若是通過以上的4個步驟,尚未解析成功,那麼會進行以下步驟:
5.操做系統就會查找NetBIOS name Cache(NetBIOS名稱緩存,就存在客戶端電腦中的),那這個緩存有什麼東西呢?凡是最近一段時間內和我成功通信的計算機的計算機名和Ip地址,就都會存在這個緩存裏面。什麼狀況下該步能解析成功呢?就是該名稱正好是幾分鐘前和我成功通訊過,那麼這一步就能夠成功解析。

6.若是第5步也沒有成功,那會查詢WINS 服務器(是NETBIOS名稱和IP地址對應的服務器)

7.若是第6步也沒有查詢成功,那麼客戶端就要進行廣播查找

8.若是第7步也沒有成功,那麼客戶端就讀取LMHOSTS文件(和HOSTS文件同一個目錄下,寫法也同樣)

若是第八步尚未解析成功,那麼就宣告此次解析失敗,那就沒法跟目標計算機進行通訊。只要這八步中有一步能夠解析成功,那就能夠成功和目標計算機進行通訊。
DNS也是開銷,一般瀏覽器查找一個給定域名的IP地址要花費20~120毫秒,在完成域名解析以前,瀏覽器不能從服務器加載到任何東西。那麼如何減小域名解析時間,加快頁面加載速度呢?

當客戶端DNS緩存(瀏覽器和操做系統)緩存爲空時,DNS查找的數量與要加載的Web頁面中惟一主機名的數量相同,包括頁面URL、腳本、樣式表、圖片、Flash對象等的主機名。減小主機名的 數量就能夠減小DNS查找的數量。

減小惟一主機名的數量會潛在減小頁面中並行下載的數量(HTTP 1.1規範建議從每一個主機名並行下載兩個組件,但實際上能夠多個),這樣減小主機名和並行下載的方案會產生矛盾,須要你們本身權衡。建議將組件放到至少兩個但很少於4個主機名下,減小DNS查找的同時也容許高度並行下載。

10、精簡JavaScript

精簡

精簡就是從代碼中移除沒必要要的字符以減小文件大小,下降加載的時間。代碼精簡的時候會移除沒必要要的空白字符(空格,換行、製表符),這樣整個文件的大小就變小了。

混淆

混淆是應用在源代碼上的另一種方式,它會移除註釋和空白符,同時它還會改寫代碼。在混淆的時候,函數和變量名將會被轉換成更短的字符串,這時代碼會更加精煉同時難以閱讀。一般這樣作是爲了增長對代碼進行反向工程的難度,這也同時提升了性能。

缺點:

混淆自己比較複雜,可能會引入錯誤。

須要對不能改變的符號作標記,防止JavaScript符號(譬如關鍵字、保留字)被修改。

混淆會使代碼難以閱讀,這使得在產品環境中調試問題更加困難。

在以上提到了關於用gzip之類的壓縮方式來壓縮文件,這邊說明一下,就算使用gzip等方式來壓縮文件,精簡代碼依然是有必要的。通常來講,壓縮產生的節省是高於精簡的,在生產環境中,精簡和壓縮同時使用可以最大限度的得到更多的節省。

CSS的精簡

CSS的精簡帶來的節省通常來講是小於JavaScript精簡的,由於CSS中註釋和空白相對較少。

除了移除空白、註釋以外,CSS能夠經過優化來得到更多的節省:

合併相同的類;

移除不使用的類;

使用縮寫,譬如

如何提升頁面加載速度(二)

.right {
    color: #fff;

    padding-top: 0; 

    margin: 0 10px;

    border: 1px solid #111
}
.wrong {
    color: #ffffff;

    padding-top: 0px; 

    margin-top: 0;
    margin-bottom: 0;
    margin-left: 10px;
    margin-right: 10px;


    border-color: #111;
    border-width: 1px;
    border-style: solid;
}

如何提升頁面加載速度(二)

上面.right是正確的的寫法,顏色使用縮寫,使用0代替0px,合併能夠合併的樣式。另外,在精簡的時候其實樣式最後一行的';'也是能夠省略的。

來看看精簡的例子:

以上分別是jquery-2.0.3的學習版(未精簡)和精簡版,可見精簡文件的大小比源文件小了155k,並且,在精簡版中jquery還作了混淆,譬如用e代替window等,從而得到最大的節省。

11、避免重定向

什麼是重定向?

重定向用於將用戶從一個URL從新路由到另外一個URL。

經常使用重定向的類型

301:永久重定向,主要用於當網站的域名發生變動以後,告訴搜索引擎域名已經變動了,應該把舊域名的的數據和連接數轉移到新域名下,從而不會讓網站的排名因域名變動而受到影響。

302:臨時重定向,主要實現post請求後告知瀏覽器轉移到新的URL。

304:Not Modified,主要用於當瀏覽器在其緩存中保留了組件的一個副本,同時組件已通過期了,這是瀏覽器就會生成一個條件GET請求,若是服務器的組件並無修改過,則會返回304狀態碼,同時不攜帶主體,告知瀏覽器能夠重用這個副本,減小響應大小。

重定向如何損傷性能?

當頁面發生了重定向,就會延遲整個HTML文檔的傳輸。在HTML文檔到達以前,頁面中不會呈現任何東西,也沒有任何組件會被下載。

來看一個實際例子:對於ASP.NET webform開發來講,對於新手很容易犯一個錯誤,就是把頁面的鏈接寫成服務器控件後臺代碼裏,例如用一個Button控件,在它的後臺click事件中寫上:Response.Redirect("");然而這個Button的做用只是轉移URL,這是很是低效的作法,由於點擊Button後,先發送一個Post請求給服務器,服務器處理Response.Redirect("")後就發送一個302響應給瀏覽器,瀏覽器再根據響應的URL發送GET請求。正確的作法應該是在html頁面直接使用a標籤作連接,這樣就避免了多餘的post和重定向。

重定向的應用場景

  1. 跟蹤內部流量

重定向常常用於跟蹤用戶流量的方向,當擁有一個門戶主頁的時候,同時想對用戶離開主頁後的流量進行跟蹤,這時可使用重定向。例如: 某網站主頁新聞的連接地址http://a.com/r/news,點擊該連接將產生301響應,其Location被設置爲http://news.a.com。經過分析a.com的web服務器日誌能夠得知人們離開首頁以後的去向。

咱們知道重定向是如何損傷性能的,爲了實現更好的效率,可使用Referer日誌來跟蹤內部流量去向。每一個HTTP請求都有一個Referer表示原始請求頁(除了從書籤打開或直接鍵入URL等操做),記錄下每一個請求的Referer,就避免了向用戶發送重定向,從而改善了響應時間。

  1. 跟蹤出站流量

有時連接可能將用戶帶離你的網站,在這種狀況下,使用Referer就不太現實了。

一樣也可使用重定向來解決跟蹤出站流量問題。以百度搜索爲例,百度經過將每一個連接包裝到一個302重定向來解決跟蹤的問題,例如搜索關鍵字「前端性能優化」,搜索結果中的一個URL爲https://www.baidu.com/link?url=pDjwTfa0IAf_FRBNlw1qLDtQ27YBujWp9jPN4q0QSJdNtGtDBK3ja3jyyN2CgxR5aTAywG4SI6V1NypkSyLISWjiFuFQDinhpVn4QE-uLGG&wd=&eqid=9c02bd21001c69170000000556ece297,即便搜索結果並無變,但這個字符串是動態改變的,暫時還不知道這裏起到怎樣的做用?(我的感受:字符串中包含了待訪問的網址,點擊以後會產生302重定向,將頁面轉到目標頁面(待修改,求大神們給我指正))

除了重定向外,咱們還能夠選擇使用信標(beacon)——一個HTTP請求,其URL中包含有跟蹤信息。跟蹤信息能夠從信標Web服務器的訪問日記中提取出來,信標一般是一個1px*1px的透明圖片,不過204響應更優秀,由於它更小,歷來不被緩存,並且毫不會改變瀏覽器的狀態。

12、刪除重複腳本

在團隊開發一個項目時,因爲不一樣開發者之間均可能會向頁面中添加頁面或組件,所以可能相同的腳本會被添加屢次。

重複的腳本會形成沒必要要的HTTP請求(若是沒有緩存該腳本的話),而且執行多餘的JavaScript浪費時間,還有可能形成錯誤。

如何避免重複腳本呢?

  1. 造成良好的腳本組織。重複腳本有可能出如今不一樣的腳本包含同一段腳本的狀況,有些是必要的,但有些卻不是必要的,因此須要對腳本進行一個良好的組織。

  2. 實現腳本管理器模塊。

例如:

 function insertScript($file) {
     if(hadInserted($file)) {
          return;
      }
      exeInsert($file);

      if(hasDependencies($file)) {

          $deps = getDependencies($file);

         foreach ($deps as $script) {
             insertScript($script);
         }

         echo "<script type='text/javascript' src='".getVersion($file)."'></script>";

     }
 }

先檢查是否插入過,若是插入過則返回。若是該腳本依賴其它腳本,則被依賴的腳本也會被插入。最後腳本被傳送到頁面,getVersion會檢查腳本並返回追加了對應版本號的文件名,這樣若是腳本的版本變化了,那麼之前瀏覽器緩存的就會失效。

十3、配置ETag

之前瀏覽器緩存的就會失效。

什麼是ETag?

實體標籤(EntityTag)是惟一標識了一個組件的一個特定版本的字符串,是web服務器用於確認緩存組件的有效性的一種機制,一般可使用組件的某些屬性來構造它。

條件GET請求

若是組件過時了,瀏覽器在重用它以前必須首先檢查它是否有效。瀏覽器將發送一個條件GET請求到服務器,服務器判斷緩存還有效,則發送一個304響應,告訴瀏覽器能夠重用緩存組件。

那麼服務器是根據什麼判斷緩存是否還有效呢?有兩種方式:

ETag(實體標籤);

最新修改日期;

最新修改日期

原始服務器經過Last-Modified響應頭來返回組件的最新修改日期。

舉個栗子:

當咱們不帶緩存訪問www.google.com.hk的時候,咱們須要下載google的logo,這時會發送這樣一個HTTP請求:

Request:

GET googlelogo_color_272x92dp.png HTTP 1.1

Host: www.google.com.hk

Response:

HTTP 1.1 200 OK

Last-Modified:Fri, 04 Sep 2015 22:33:08 GMT

如何提升頁面加載速度(二)

當須要再次訪問相同組件的時候,同時緩存已通過期,瀏覽器會發送以下條件GET請求:

Request:

GET googlelogo_color_272x92dp.png HTTP 1.1

If-Modified-Since:Fri, 04 Sep 2015 22:33:08 GMT

Host: www.google.com.hk

Response:

HTTP 1.1 304 Not Modified

如何提升頁面加載速度(二)

實體標籤

ETag提供了另一種方式,用於檢測瀏覽器緩存中的組件與原始服務器上的組件是否匹配。摘抄自書上的例子:

不帶緩存的請求:

Request:

GET /i/yahoo/gif HTTP 1.1

Host: us.yimg.com

Response:

HTTP 1.1 200 OK

Last-Modified:Tue,12 Dec 200603:03:59 GMT

ETag:」10c24bc-4ab-457elc1f「


  再次請求相同組件:

Request:

GET /i/yahoo/gif HTTP 1.1

Host: us.yimg.com

If-Modified-Since:Tue,12 Dec 200603:03:59 GMT

If-None-Match:」10c24bc-4ab-457elc1f「

Response:

HTTP 1.1 304 Not Midified

爲何要引入ETag?

ETag主要是爲了解決Last-Modified沒法解決的一些問題:

  1. 一些文件也許會週期性的更改,可是他的內容並不改變(僅僅改變的修改時間),這個時候咱們並不但願客戶端認爲這個文件被修改了,而從新GET;

  2. 某些文件修改很是頻繁,好比在秒如下的時間內進行修改,(比方說1s內修改了N次),If-Modified-Since能檢查到的粒度是s級的,這種修改沒法判斷(或者說UNIX記錄MTIME只能精確到秒);

  3. 某些服務器不能精確的獲得文件的最後修改時間。

ETag帶來的問題

ETag的問題在於一般使用某些屬性來構造它,有些屬性對於特定的部署了網站的服務器來講是惟一的。當使用集羣服務器的時候,瀏覽器從一臺服務器上獲取了原始組件,以後又向另一臺不一樣的服務器發起條件GET請求,ETag就會出現不匹配的情況。例如:使用inode-size-timestamp來生成ETag,文件系統使用inode存儲文件類型、全部者、組和訪問模式等信息,在多臺服務器上,就算文件大小、權限、時間戳等都相同,inode也是不一樣的。

最佳實踐

  1. 若是使用Last-Modified不會出現任何問題,能夠直接移除ETag,google的搜索首頁則沒有使用ETag。

  2. 肯定要使用ETag,在配置ETag的值的時候,移除可能影響到組件集羣服務器驗證的屬性,例如使用size-timestamp來生成時間戳。

十4、使Ajax可緩存

維基百科中這樣定義Ajax:

AJAX即「Asynchronous JavaScript and XML」(異步的JavaScript與XML技術),指的是一套綜合了多項技術的瀏覽器端網頁開發技術。Ajax的概念由傑西·詹姆士·賈瑞特所提出。

傳統的Web應用容許用戶端填寫表單(form),當提交表單時就向Web服務器發送一個請求。服務器接收並處理傳來的表單,而後送回一個新的網頁,但這個作法浪費了許多帶寬,由於在先後兩個頁面中的大部分HTML碼每每是相同的。因爲每次應用的溝通都須要向服務器發送請求,應用的迴應時間依賴於服務器的迴應時間。這致使了用戶界面的迴應比本機應用慢得多。

與此不一樣,AJAX應用能夠僅向服務器發送並取回必須的數據,並在客戶端採用JavaScript處理來自服務器的迴應。由於在服務器和瀏覽器之間交換的數據大量減小(大約只有原來的5%)[來源請求],服務器迴應更快了。同時,不少的處理工做能夠在發出請求的客戶端機器上完成,所以Web服務器的負荷也減小了。

相似於DHTML或LAMP,AJAX不是指一種單一的技術,而是有機地利用了一系列相關的技術。雖然其名稱包含XML,但實際上數據格式能夠由JSON代替,進一步減小數據量,造成所謂的AJAJ。而客戶端與服務器也並不須要異步。一些基於AJAX的「派生/合成」式(derivative/composite)的技術也正在出現,如AFLAX。

Ajax的目地是爲突破web本質的開始—中止交互方式,向用戶顯示一個白屏後重繪整個頁面不是一種好的用戶體驗。

異步與即時

Ajax的一個明顯的有點就是向用戶提供了即時反饋,由於它異步的從後端web服務器請求信息。

用戶是否須要等待的關鍵因素在於Ajax請求是被動的仍是主動的。被動請求是爲了未來來使用而預先發起的,主動請求是基於用戶當前的操做而發起的

什麼樣的AJAX請求能夠被緩存?

POST的請求,是不能夠在客戶端緩存的,每次請求都須要發送給服務器進行處理,每次都會返回狀態碼200。(能夠在服務器端對數據進行緩存,以便提升處理速度)

GET的請求,是能夠(並且默認)在客戶端進行緩存的,除非指定了不一樣的地址,不然同一個地址的AJAX請求,不會重複在服務器執行,而是返回304。

Ajax請求使用緩存

在進行Ajax請求的時候,能夠選擇儘可能使用get方法,這樣可使用客戶端的緩存,提升請求速度。

做者:MarcoHan
出處:
http://www.cnblogs.com/MarcoHan/p/5295398.html


 

如何提升頁面加載速度(二)

識別圖中二維碼,領取python全套視頻資料

相關文章
相關標籤/搜索