于吉吉的技術博客

          建造高性能門戶網(wǎng)

            BlogJava :: 首頁 :: 新隨筆 :: 聯(lián)系 :: 聚合  :: 管理 ::
            65 隨筆 :: 6 文章 :: 149 評論 :: 0 Trackbacks
          記得問過身邊的一些開發(fā)工程師(非前端)緩存要分幾個層次,從哪里做起,答案很多,比如反向代理緩存,DNS緩存,memcached,數(shù)據(jù)庫緩存等等,確實很完整,不過好像漏掉了我們用戶跟我們聯(lián)通的最根本的工具瀏覽器,確實好似很少有人把用戶的瀏覽器當作是web站點的組成部分來看待

          緩存協(xié)商
          現(xiàn)在我們需要將用戶的瀏覽器也納入我們構(gòu)建網(wǎng)站各個緩存層次中的其中一個重要層次,網(wǎng)站信息和內(nèi)容在由web服務器生成,而將這些信息和內(nèi)容作為一段二進制的文件作為本地緩存文件存放在用戶的瀏覽器,是兩個獨立個體共同完成的任務,所以兩者之間需要一種溝通的機制,也就是HTTP的緩存協(xié)商

          Last-Modified和If-Modified-Since協(xié)商
          Last-Modified和If-Modified-Since分別位于響應頭信息和請求頭信息中,都是記錄請求的頁面最后的修改時間
          在第一次訪問web服務器會返回200狀態(tài),并在瀏覽器的響應頭Last-Modified上寫上此頁面最后修改的時間戳
          使用firebug進行查看




          在再次訪問時,瀏覽器會把第一次返回Last-Modified的時間戳記錄到If-Modified-Since,并作為請求頭信息發(fā)送到服務區(qū),web服務器會通過If-Modified-Since上的時間戳來判斷用戶的頁面是否是最新的,如果不是最新的,則返回新的頁面并修改響應頭Last-Modified時間戳給用戶,如果判斷是最新的,則返回304狀態(tài)并告訴瀏覽器本地的cache頁面是最新的,瀏覽器可以直接加載本地頁面,這樣可以減少網(wǎng)絡上傳輸?shù)臄?shù)據(jù),并且也減少服務器的負擔。




          這里需要注意的是,HTTP協(xié)議中規(guī)定使用的是GMT時間(格林威治),而我們國家使用的是GMT+8的時間,所以我們從firebug中看到的時間是比我們早了8個小時,不過這沒關系,它并不影響到緩存的對比工作

          Etag協(xié)商
          如果我們的一個文件存放在多臺web服務器上,用戶的請求在這些服務器上之間輪詢,實現(xiàn)負載均衡,那些這個文件在各臺web服務器的最后修改時間很可能是不一樣,這樣用戶每次請求到的web服務器都可能是不同,Last-Modified和If-Modified-Since則無法對應,導致每次都需要重新獲取內(nèi)容,這時候采用直接標記內(nèi)容的Etag算法,就可以避免上述的問題
          Etag是由Web服務器生成的,如Apache為一個靜態(tài)文件新增響應頭為
          Etag    "e3af4060-5-472bedf076880"
          瀏覽器獲得這個Etag后,便會在下次請求該頁面時,在HTTP的請求頭附上If-None-Match    "e3af4060-5-472bedf076880",與web服務器上的Etag值相比較,如果是相同的話,便返回304狀態(tài),不同則重新返回新頁面信息



          當然啦,這里要指出瀏覽器緩存只針對HTML和HTM進行而動態(tài)文件則無法緩存,瀏覽器并不在乎你的文件是否動態(tài)還是靜態(tài),瀏覽器只認準和web服務器通訊的HTTP協(xié)商,所以只要動態(tài)頁面HTTP頭信息包含的緩存協(xié)商信息,動態(tài)內(nèi)容也是可以一樣被瀏覽器進行緩存的,和靜態(tài)頁面無什么兩樣,只是你確定你需要這樣做,讓動態(tài)頁面被緩存?
          以下是對一個jsp進行寫頭信息

          response.addHeader("Last-Modified",DateUtl.format(new java.util.Date())+" GMT");
          response.addHeader(
          "ETag","e3af4060-5-472bedf076880");


          同樣,這樣兩個設置會寫入瀏覽器的響應頭,這個動態(tài)文件跟靜態(tài)文件一樣會被瀏覽器緩存

          Expires
          Expires標記告訴瀏覽器該頁面何時過期,并且在此過期前不需要再訪問web服務器,直接使用本地的緩存文件即可,這樣請求響應頭都不需要,確實節(jié)省了帶寬和服務器的開銷,但是就算頁面在web服務器上更新后在Expires過期前也不會出現(xiàn)在用戶面前,究竟是否應該在于我們的實際應用和取舍
          下面是Apache中提供的mod_expires模塊
          <IfModule mod_expires.c>
          ExpiresActive On
          ExpiresDefault A600
          ExpiresByType image/x-icon A2592000
          ExpiresByType application/x-javascript A604800
          ExpiresByType text/css A604800
          ExpiresByType image/gif A2592000
          ExpiresByType image/png A2592000
          ExpiresByType image/jpeg A2592000
          ExpiresByType text/plain A86400
          ExpiresByType application/x-shockwave-flash A2592000
          ExpiresByType video/x-flv A2592000
          ExpiresByType application/pdf A2592000
          ExpiresByType text/html A600
          </IfModule>

          或者
          <IfModule mod_expires.c>
          ExpiresActive On
          ExpiresDefault "access plus 12 hours"
          ExpiresByType text/html "access plus 3 days"
          ExpiresByType text/plain "access plus 3 days"
          ExpiresByType text/css  "access plus 7 days"
          ExpiresByType image/gif "access plus 30 days"
          ExpiresByType image/png "access plus 30 days"
          ExpiresByType image/jpeg "access plus 30 days"
          ExpiresByType image/x-icon "access plus 30 days"
          ExpiresByType video/x-flv  "access plus 30 days"
          ExpiresByType application/x-shockwave-flash "access plus 30 days"
          </IfModule>


          我們在jsp文件也可以指定expires的過期時間

          response.addHeader("Expires",DateUtl.format(new java.util.Date())+" GMT");



          當然啦,你再刷新也不會返回304狀態(tài),因為瀏覽器已經(jīng)不用想web服務器發(fā)出請求

          ----------------------------------------

          by 陳于喆
          Mail: chenyz@corp.netease.com


          參考文章
          郭欣 《構(gòu)建高新能web站點》
          中文版 《HTTP協(xié)議(RFC2616)》
          posted on 2010-08-25 00:03 陳于喆 閱讀(6443) 評論(0)  編輯  收藏 所屬分類: web開發(fā)緩存
          主站蜘蛛池模板: 麻城市| 江源县| 全椒县| 大英县| 油尖旺区| 华池县| 孟连| 乌兰浩特市| 内江市| 延安市| 刚察县| 科技| 孟连| 和硕县| 和顺县| 疏附县| 从江县| 平南县| 分宜县| 东乌珠穆沁旗| 万源市| 成武县| 青田县| 叶城县| 新野县| 石楼县| 电白县| 方山县| 沙洋县| 林周县| 介休市| 凤阳县| 册亨县| 台东县| 崇仁县| 郁南县| 吴川市| 泽普县| 南郑县| 甘肃省| 渭南市|