Web前段優化,提升加載速度 css

前言: javascript

在一樣的網絡環境下,兩個一樣能知足你的需求的網站,一個「Duang」的一下就加載出來了,一個糾結了半天才出來,你會選擇哪一個?研究代表:用戶最滿意的打開網頁時間是2-5秒,若是等待超過10秒,99%的用戶會關閉這個網頁。也許這樣講,各位還不會有太多感觸,接下來我列舉一組數據:Google網站訪問速度每慢400ms就致使用戶搜索請 求降低0.59%;Amazon每增長100ms網站延遲將致使收入降低1%;雅虎若是有400ms延遲會致使流量降低5-9%。網站的加載速度嚴重影響了用戶體驗,也決定了這個網站的生死存亡。css

可能有人會說:網站的性能是後端工程師的事情,與前端並沒有多大關係。我只能說,too young too simple。事實上,只有10%~20%的最終用戶響應時間是用在從Web服務器獲取HTML文檔並傳送到瀏覽器的,那剩餘的時間去哪兒了?來瞄一下性能黃金法則html

只有10%~20%的最終用戶響應時間花在了下載HTML文檔上。其他的80%~90%時間花在了下載頁面中的全部組件上。前端

接下來咱們將研究一下前端攻城獅如何來提升頁面的加載速度。java

 

1、減小HTTP請求node

上面說到80%~90%時間花在了下載頁面中的全部組件進行的HTTP請求上。所以,改善響應時間最簡單的途徑就是減小HTTP請求的數量。jquery

圖片地圖:linux

假設導航欄上有五幅圖片,點擊每張圖片都會進入一個連接,這樣五張導航的圖片在加載時會產生5個HTTP請求。然而,使用一個圖片地圖能夠提升效率,這樣就只須要一個HTTP請求。web

服務器端圖片地圖:將全部點擊提交到同一個url,同時提交用戶點擊的x、y座標,服務器端根據座標映射響應chrome

客戶端圖片地圖:直接將點擊映射到操做

複製代碼
<img src="planets.jpg" border="0" usemap="#planetmap" alt="Planets" /> <map name="planetmap" id="planetmap"> <area shape="rect" coords="180,139,14" href ="venus.html" alt="Venus" /> <area shape="rect" coords="129,161,10" href ="mercur.html" alt="Mercury" /> <area shape="rect" coords="0,0,110,260" href ="sun.html" alt="Sun" /> <area shape="rect" coords="140,0,110,260" href ="star.html" alt="Sun" /> </map>
複製代碼

使用圖片地圖的缺點:指定座標區域時,矩形或圓形比較容易指定,而其它形狀手工指定比較難

CSS Sprites

CSS Sprites直譯過來就是CSS精靈,可是這種翻譯顯然是不夠的,其實就是經過將多個圖片融合到一副圖裏面,而後經過CSS的一些技術佈局到網頁上。特別是圖片特別多的網站,若是能用css sprites下降圖片數量,帶來的將是速度的提高。

複製代碼
<div> <span id="image1" class="nav"></span> <span id="image2" class="nav"></span> <span id="image3" class="nav"></span> <span id="image4" class="nav"></span> <span id="image5" class="nav"></span> </div>
複製代碼
複製代碼
.nav { width: 50px; height: 50px; display: inline-block; border: 1px solid #000; background-image: url('E:/1.png'); } #image1 { background-position: 0 0; } #image2 { background-position: -95px 0; } #image3 { background-position: -185px 0; } #image4 { background-position: -275px 0; } #image5 { background-position: -366px -3px; }
複製代碼

運行結果:

PS:使用CSS Sprites還有可能下降下載量,可能你們會認爲合併後的圖片會比分離圖片的總和要大,由於還有可能會附加空白區域。實際上,合併後的圖片會比分離的圖片總和要小,由於它下降了圖片自身的開銷,譬如顏色表、格式信息等。

字體圖標

在能夠大量使用字體圖標的地方咱們能夠儘量使用字體圖標,字體圖標能夠減小不少圖片的使用,從而減小http請求,字體圖標還能夠經過CSS來設置顏色、大小等樣式,何樂而不爲。

合併腳本 和樣式表

將多個樣式表或者腳本文件合併到一個文件中,能夠減小HTTP請求的數量從而縮短效應時間。

然而合併全部文件對許多人尤爲是編寫模塊化代碼的人來講是不能忍的,並且合併全部的樣式文件或者腳本文件可能會致使在一個頁面加載時加載了多於本身所須要的樣式或者腳本,對於只訪問該網站一個(或幾個)頁面的人來講反而增長了下載量,因此你們應該本身權衡利弊。

 

2、使用CDN

若是應用程序web服務器離用戶更近,那麼一個HTTP請求的響應時間將縮短。另外一方面,若是組件web服務器離用戶更近,則多個HTTP請求的響應時間將縮短。

 CDN(內容發佈網絡)是一組分佈在多個不一樣地理位置的Web服務器,用於更加有效地向用戶發佈內容。在優化性能時,向特定用戶發佈內容的服務器的選擇基於對網絡慕課擁堵的測量。例如,CDN可能選擇網絡階躍數最小的服務器,或者具備最短響應時間的服務器。

CDN還能夠進行數據備份、擴展存儲能力,進行緩存,同時有助於緩和Web流量峯值壓力。

CDN的缺點:

一、響應時間可能會受到其餘網站流量的影響。CDN服務提供商在其全部客戶之間共享Web服務器組。

二、若是CDN服務質量降低了,那麼你的工做質量也將降低

三、沒法直接控制組件服務器

 

3、添加Expires頭

頁面的初次訪問者會進行不少HTTP請求,可是經過使用一個長久的Expires頭,可使這些組件被緩存,下次訪問的時候,就能夠減小沒必要要的HTPP請求,從而提升加載速度。

Web服務器經過Expires頭告訴客戶端可使用一個組件的當前副本,直到指定的時間爲止。例如:

Expires: Fri, 18 Mar 2016 07:41:53 GMT

Expires缺點: 它要求服務器和客戶端時鐘嚴格同步;過時日期須要常常檢查

HTTP1.1中引入Cache-Control來克服Expires頭的限制,使用max-age指定組件被緩存多久。

Cache-Control: max-age=12345600

若同時制定Cache-Control和Expires,則max-age將覆蓋Expires頭

 

4、壓縮組件

從HTTP1.1開始,Web客戶端能夠經過HTTP請求中的Accept-Encoding頭來表示對壓縮的支持

Accept-Encoding: gzip,deflate

若是Web服務器看到請求中有這個頭,就會使用客戶端列出來的方法中的一種來進行壓縮。Web服務器經過響應中的Content-Encoding來通知 Web客戶端。

Content-Encoding: gzip

代理緩存

當瀏覽器經過代理來發送請求時,狀況會不同。假設針對某個URL發送到代理的第一個請求來自於一個不支持gzip的瀏覽器。這是代理的第一個請求,緩存爲空。代理將請求轉發給服務器。此時響應是未壓縮的,代理緩存同時發送給瀏覽器。如今,假設到達代理的請求是同一個url,來自於一個支持gzip的瀏覽器。代理會使用緩存中未壓縮的內容進行響應,從而失去了壓縮的機會。相反,若是第一個瀏覽器支持gzip,第二個不支持,大家代理緩存中的壓縮版本將會提供給後續的瀏覽器,而無論它們是否支持gzip。

解決辦法:在web服務器的響應中添加vary頭Web服務器能夠告訴代理根據一個或多個請求頭來改變緩存的響應。由於壓縮的決定是基於Accept-Encoding請求頭的,所以須要在vary響應頭中包含Accept-Encoding。

vary: Accept-Encoding
 

5、將樣式表放在頭部

首先說明一下,將樣式表放在頭部對於實際頁面加載的時間並不能形成太大影響,可是這會減小頁面首屏出現的時間,使頁面內容逐步呈現,改善用戶體驗,防止「白屏」。

咱們老是但願頁面可以儘快顯示內容,爲用戶提供可視化的回饋,這對網速慢的用戶來講是很重要的。

將樣式表放在文檔底部會阻止瀏覽器中的內容逐步出現。爲了不當樣式變化時重繪頁面元素,瀏覽器會阻塞內容逐步呈現,形成「白屏」。這源自瀏覽器的行爲:若是樣式表仍在加載,構建呈現樹就是一種浪費,由於全部樣式表加載解析完畢以前務虛會之任何東西

 

6、將腳本放在底部

更樣式表相同,腳本放在底部對於實際頁面加載的時間並不能形成太大影響,可是這會減小頁面首屏出現的時間,使頁面內容逐步呈現。

js的下載和執行會阻塞Dom樹的構建(嚴謹地說是中斷了Dom樹的更新),因此script標籤放在首屏範圍內的HTML代碼段裏會截斷首屏的內容。

下載腳本時並行下載是被禁用的——即便使用了不一樣的主機名,也不會啓用其餘的下載。由於腳本可能修改頁面內容,所以瀏覽器會等待;另外,也是爲了保證腳本可以按照正確的順序執行,由於後面的腳本可能與前面的腳本存在依賴關係,不按照順序執行可能會產生錯誤。

 

7、避免CSS表達式

CSS表達式是動態設置CSS屬性的一種強大而且危險的方式,它受到了IE5以及以後版本、IE8以前版本的支持。

p { width: expression(func(),document.body.clientWidth > 400 ? "400px" : "auto"); height: 80px; border: 1px solid #f00; }
複製代碼
<p><span></span></p> <p><span></span></p> <p><span></span></p> <p><span></span></p> <p><span></span></p> <script> var n = 0; function func() { n++;  alert();  console.log(n); } </script>
複製代碼

鼠標移動了幾回,函數的運行次數垂手可得的達到了幾千次,危險性顯而易見。

如何解決:

一次性表達式:

p { width: expression(func(this)); height: 80px; border: 1px solid #f00; }
複製代碼
<p><span></span></p> <p><span></span></p> <p><span></span></p> <p><span></span></p> <p><span></span></p> <script> var n = 0; function func(elem) { n++; elem.style.width 400 ? '400px' : "auto"; console.log(n); } </script>
複製代碼

事件處理機制

用js事件處理機制來動態改變元素的樣式,使函數運行次數在可控範圍以內。

 

8、使用外部的JavaScript和CSS

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

影響因素:

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

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

加載後下載

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

複製代碼
 1   function doOnload() {  2 setTimeout("downloadFile()",1000);  3 }  4  5 window.onload = doOnload;  6  7 function downloadFile() {  8 downloadCss("http://abc.com/css/a.css");  9 downloadJS("http://abc.com/js/a.js"); 10 } 11 12 function downloadCss(url) { 13 var ele = document.createElement('link'); 14 ele.rel = "stylesheet"; 15 ele.type = "text/css"; 16 ele.href = url; 17 18 document.body.appendChild(ele); 19 } 20 21 function downloadJS(url) { 22 var ele = document.createElement('script'); 23 ele.src = url; 24 document.body.appendChild(ele); 25 }
複製代碼

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

 

 9、減小DNS查找

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

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

域名解析是頁面加載的第一步,那麼域名是如何解析的呢?以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:\Windows\System32\drivers\etc),看看這裏面有沒有該域名對應的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,就避免了向用戶發送重定向,從而改善了響應時間。

2. 跟蹤出站流量

有時連接可能將用戶帶離你的網站,在這種狀況下,使用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. 實現腳本管理器模塊。

例如:

複製代碼
 1  function insertScript($file) {  2 if(hadInserted($file)) {  3 return;  4 }  5 exeInsert($file);  6  7 if(hasDependencies($file)) {  8  9 $deps = getDependencies($file); 10 11 foreach ($deps as $script) { 12 insertScript($script); 13 } 14 15 echo "<script type='text/javascript' src='".getVersion($file)."'></script>"; 16 17 } 18 }
複製代碼

先檢查是否插入過,若是插入過則返回。若是該腳本依賴其它腳本,則被依賴的腳本也會被插入。最後腳本被傳送到頁面,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方法,這樣可使用客戶端的緩存,提升請求速度。

相關文章
相關標籤/搜索