者:kevinylzhao,騰訊音樂前端開發(fā)工程師
瀏覽器緩存策略對于前端開發(fā)同學(xué)來說不陌生,大家都有一定的了解,但如果沒有系統(tǒng)的歸納總結(jié),可能三言兩語很難說明白,甚至說錯,尤其在面試過程中感觸頗深,很多候選人對這類基礎(chǔ)知識竟然都是一知半解,說出幾個概念就沒了,所以重新歸納總結(jié)下,溫故而知新。
瀏覽器緩存一般分為兩類:強緩存(也稱本地緩存)和協(xié)商緩存(也稱弱緩存)。
瀏覽器發(fā)送請求前,會先去緩存里查看是否命中強緩存,如果命中,則直接從緩存中讀取資源,不會發(fā)送請求到服務(wù)器。否則,進入下一步。
當(dāng)強緩存沒有命中時,瀏覽器一定會向服務(wù)器發(fā)起請求。服務(wù)器會根據(jù) Request Header 中的一些字段來判斷是否命中協(xié)商緩存。如果命中,服務(wù)器會返回 304 響應(yīng),但是不會攜帶任何響應(yīng)實體,只是告訴瀏覽器可以直接從瀏覽器緩存中獲取這個資源。如果本地緩存和協(xié)商緩存都沒有命中,則從直接從服務(wù)器加載資源。
按照本地緩存階段和協(xié)商緩存階段分類:
上述代碼的作用是告訴瀏覽器當(dāng)前頁面不被緩存,事實上這種禁用緩存的形式用處很有限:
a. 僅有 IE 才能識別這段 meta 標(biāo)簽含義,其它主流瀏覽器僅識別“Cache-Control: no-store”的 meta 標(biāo)簽。
b. 在 IE 中識別到該 meta 標(biāo)簽含義,并不一定會在請求字段加上 Pragma,但的確會讓當(dāng)前頁面每次都發(fā)新請求(僅限頁面,頁面上的資源則不受影響)。
在 HTTP 請求和響應(yīng)的消息報頭中,常見的與緩存有關(guān)的消息報頭有:
上圖中只是常用的消息報頭,下面來看下不同字段之間的關(guān)系和區(qū)別:
a. Last-Modified 標(biāo)注的最后修改只能精確到秒級,如果某些文件在 1 秒鐘以內(nèi),被修改多次的話,它將不能準確標(biāo)注文件的新鮮度;
b. 某些文件也許會周期性的更改,但是它的內(nèi)容并不改變(僅僅改變的修改時間),但 Last-Modified 卻改變了,導(dǎo)致文件沒法使用緩存;
c. 有可能存在服務(wù)器沒有準確獲取文件修改時間,或者與代理服務(wù)器時間不一致等情形。
瀏覽器可以在內(nèi)存、硬盤中開辟一個空間用于保存請求資源副本。我們經(jīng)常調(diào)試時在 DevTools Network 里看到 Memory Cache(內(nèi)存緩存)和 Disk Cache(硬盤緩存),指的就是緩存所在的位置。請求一個資源時,會按照優(yōu)先級(Service Worker -> Memory Cache -> Disk Cache -> Push Cache)依次查找緩存,如果命中則使用緩存,否則發(fā)起請求。這里先介紹 Memory Cache 和 Disk Cache。
200 from memory cache
表示不訪問服務(wù)器,直接從內(nèi)存中讀取緩存。因為緩存的資源保存在內(nèi)存中,所以讀取速度較快,但是關(guān)閉進程后,緩存資源也會隨之銷毀,一般來說,系統(tǒng)不會給內(nèi)存分配較大的容量,因此內(nèi)存緩存一般用于存儲較小文件。同時內(nèi)存緩存在有時效性要求的場景下也很有用(比如瀏覽器的隱私模式)。
200 from disk cache
表示不訪問服務(wù)器,直接從硬盤中讀取緩存。與內(nèi)存相比,硬盤的讀取速度相對較慢,但硬盤緩存持續(xù)的時間更長,關(guān)閉進程之后,緩存的資源仍然存在。由于硬盤的容量較大,因此一般用于存儲大文件。
下圖可清晰看出差別:
200 from prefetch cache
在 preload 或 prefetch 的資源加載時,兩者也是均存儲在 http cache,當(dāng)資源加載完成后,如果資源是可以被緩存的,那么其被存儲在 http cache 中等待后續(xù)使用;如果資源不可被緩存,那么其在被使用前均存儲在 memory cache。
CDN Cache
以騰訊 CDN 為例:X-Cache-Lookup:Hit From MemCache 表示命中 CDN 節(jié)點的內(nèi)存;X-Cache-Lookup:Hit From Disktank 表示命中 CDN 節(jié)點的磁盤;X-Cache-Lookup:Hit From Upstream 表示沒有命中 CDN。
從上圖能感受到整個流程,比如常見兩種刷新場景:
IndexedDB 就是瀏覽器提供的本地數(shù)據(jù)庫,能夠在客戶端存儲可觀數(shù)量的結(jié)構(gòu)化數(shù)據(jù),并且在這些數(shù)據(jù)上使用索引進行高性能檢索的 API。
異步 API 方法調(diào)用完后會立即返回,而不會阻塞調(diào)用線程。要異步訪問數(shù)據(jù)庫,要調(diào)用 window 對象 indexedDB 屬性的 open() 方法。該方法返回一個 IDBRequest 對象 (IDBOpenDBRequest);異步操作通過在 IDBRequest 對象上觸發(fā)事件來和調(diào)用程序進行通信。
常用異步 API 如下:
在 16 年曾基于 IndexDB 做過一整套緩存策略,有不錯的優(yōu)化效果:
SW 從 2014 年提出的草案到現(xiàn)在已經(jīng)發(fā)展很成熟了,基于 SW 做離線緩存,讓用戶能夠進行離線體驗,消息推送體驗,離線緩存能力涉及到 Cache 和 CacheStorage 的概念,篇幅有限,不展開了。
localStorage 屬性允許你訪問一個 Document 源(origin)的對象 Storage 用于存儲當(dāng)前源的數(shù)據(jù),除非用戶人為清除(調(diào)用 localStorage api 或者清除瀏覽器數(shù)據(jù)), 否則存儲在 localStorage 的數(shù)據(jù)將被長期保留。
sessionStorage 屬性允許你訪問一個 session Storage 對象,用于存儲當(dāng)前會話的數(shù)據(jù),存儲在 sessionStorage 里面的數(shù)據(jù)在頁面會話結(jié)束時會被清除。頁面會話在瀏覽器打開期間一直保持,并且重新加載或恢復(fù)頁面仍會保持原來的頁面會話。
通過了解瀏覽器各種緩存機制和存儲能力特點,結(jié)合業(yè)務(wù)制定合適的緩存策略,善用緩存是基本功,可以用于時常審查負責(zé)的業(yè)務(wù),可能就會發(fā)現(xiàn)個別業(yè)務(wù)并沒有運用到位,共勉。
決這個問題, 有兩個可用的方法, 使用HTML meta標(biāo)簽和使用HTTP頭。
1.使用HTML meta標(biāo)簽
阻止緩存頁面最常用的方法是使用HTML meta標(biāo)簽:
<meta http-equiv="expires" content="Wed,15 Jul 2023 07:30:28 GMT"/>
<meta http-equiv="pragma" content="no-cache"/>
已經(jīng)傳遞到Expires meta標(biāo)簽的插入日期告訴瀏覽器,緩存的頁面拷貝總是過期的。遇到此標(biāo)簽后, 瀏覽器通常不會緩存頁面。雖然Pragma: no-cache meta標(biāo)簽并不是保證,但它卻是大多數(shù)網(wǎng)絡(luò)瀏覽器遵從的受到良好支持的約定。
2.使用HTTP頭
阻止緩存頁面更好的辦法是, 應(yīng)用header()函數(shù), 使用HTTP本身協(xié)議生成與上面等價的兩個HTML meta標(biāo)簽:
<?php
header('Expires:Wed,15 Jul 2023 07:30:28 GMT');
header('Pragma:no-cache');
?>
可更進一步, 可使用HTTP1.1的瀏覽器所支持的Cache-Control頭:
一步、按下Win+X組合鍵,喚出隱藏菜單,點擊里面的“設(shè)備管理器”。
第二步、在Win10設(shè)備管理器窗口,點開磁盤驅(qū)動器項目,然后對著電腦硬盤單擊鼠標(biāo)右鍵,點擊選擇菜單中的“屬性”。
第三步、在硬盤屬性對話框,點擊選擇“策略”選項卡,將“啟用此設(shè)備上的寫入緩存”更具個人需要設(shè)置成勾選,或取消勾選后,點擊底部“確定”即可。
溫馨提示:勾選“啟用此設(shè)備上的寫入緩存”是啟用該功能,取消勾選是關(guān)閉。
*請認真填寫需求信息,我們會在24小時內(nèi)與您取得聯(lián)系。