PHP客戶端緩存詳解

        一、什麼是”Last-Modified”? 
        在瀏覽器第一次請求某一個URL時,服務器端的返回狀態會是200,內容是你請求的資源,同時有一個Last-Modified的屬性標記此文件在服務期端最後被修改的時間,格式類似這樣: 
        Last-Modified: Fri, 12 May 2006 18:53:33 GMT 
        客戶端第二次請求此URL時,根據 HTTP 協議的規定,瀏覽器會向服務器傳送 If-Modified-Since 報頭,詢問該時間之後文件是否有被修改過: 
        If-Modified-Since: Fri, 12 May 2006 18:53:33 GMT 
        如果服務器端的資源沒有變化,則自動返回 HTTP 304 (Not Changed.)狀態碼,內容爲空,這樣就節省了傳輸數據量。當服務器端代碼發生改變或者重啓服務器時,則重新發出資源,返回和第一次請求時類似。從而保證不向客戶端重複發出資源,也保證當服務器有變化時,客戶端能夠得到最新的資源。 
      注意:

         1、對於一些圖像,css,js等靜態文件資源,配置好了的apache服務器會理解這些If-Modified-Since請求頭標,將頭標裏的時間和文件的最後修改時間進行比較並作出響應,如果二者相等則發送一個304 Not Modfied來告訴客戶端所請求資源並未修改讓客戶端放心使用緩存中的資源,否則的話會重新發送一個新的資源和新的Last-Modified的頭標。

          2、但是對於一個動態的PHP腳本,我們即使在腳本加入了header('Last Modified: '.$time)來發送一個Last Modified響應頭標,當客戶端附帶'If-Modified-Since'在次請求時apache服務器不會進行處理,這需要我們自己用$_SERVER['HTTP_IF_MODIFIED_SINCE']來獲取'If-Modified-Since'的值自己來進行判斷處理。

       

<?php

//設置 Last-Modified  $mktime爲所設定的時間
header( 'Last-Modified: ' . gmdate ( 'r' , $mktime ) . ' GMT' );  
//再次請求時獲取Last-Modified值
$again = $_SERVER['HTTP_IF_MODIFIED_SINCE'];
echo "".$again;

?>


        二、什麼是”Etag”? 
        HTTP 協議規格說明定義ETag爲“被請求變量的實體值” 。另一種說法是,ETag是一個可以與Web資源關聯的記號(token)。典型的Web資源可以一個Web頁,但也可能是JSON或XML文檔。服務器單獨負責判斷記號是什麼及其含義,並在HTTP響應頭中將其傳送到客戶端,以下是服務器端返回的格式: 
        ETag: "50b1c1d4f775c61:df3" 
        客戶端的查詢更新格式是這樣的: 
        If-None-Match: W/"50b1c1d4f775c61:df3" 
        如果ETag沒改變,則返回狀態304然後不返回,這也和Last-Modified一樣。本人測試Etag主要在斷點下載時比較有用。

  

<?php

//設置 ETag  $etab爲所設定的標誌
$etab = "yuanyanbing";  
header( 'ETag: '.$etab );
//再次請求時獲取ETag:值
$againb = $_SERVER['HTTP_IF_NONE_MATCH'];
echo "".$againb;

?>


     Last-Modified和Etags如何幫助提高性能?
        聰明的開發者會把Last-Modified 和ETags請求的http報頭一起使用,這樣可利用客戶端(例如瀏覽器)的緩存。因爲服務器首先產生 Last-Modified/Etag標記,服務器可在稍後使用它來判斷頁面是否已經被修改。本質上,客戶端通過將該記號傳回服務器要求服務器驗證其(客戶端)緩存。 
        過程如下:
                1. 客戶端請求一個頁面(A)。 
                2. 服務器返回頁面A,並在給A加上一個Last-Modified/ETag。 
                3. 客戶端展現該頁面,並將頁面連同Last-Modified/ETag一起緩存。 
                4. 客戶再次請求頁面A,並將上次請求時服務器返回的Last-Modified/ETag一起傳遞給服務器。 
                5. 服務器檢查該Last-Modified或ETag,並判斷出該頁面自上次客戶端請求之後還未被修改,直接返回響應304和一個空的響應體。

        三、Expires

Expires表明緩存何時因該過期(GMT時間),屬於HTTP 1.0 標準,通常是用來對Cache-Control的max-age的一個補充,來兼容HTTP 1.0,不贊成單獨使用Expires,因爲客戶端時間容易發生偏差。

        四、Pragma

HTTP 1.0 標準,通常是在不緩存時使用,Pragma: no-cache。


 

        五、關於Cache-Control

 

public

可以在任何地方緩存

private

內容只緩存到私有緩存中

no-cache

不能在任何地方緩存

must-revalidate

緩存必須檢查跟新版本

proxy-revalidate

代理緩存必須檢查跟新版本

max-age

內容能夠被緩存的時間

s-maxage

覆蓋共享緩存的max-age設置

 

Cache-directive 打開一個新的瀏覽器窗口 在原窗口中單擊 Enter 按鈕 刷新 單擊 Back 按鈕
public 瀏覽器呈現來自緩存的頁面 瀏覽器呈現來自緩存的頁面 瀏覽器重新發送請求到服務器 瀏覽器呈現來自緩存的頁面
private 瀏覽器重新發送請求到服務器 第一次,瀏覽器重新發送請求到服務器;此後,瀏覽器呈現來自緩存的頁面 瀏覽器重新發送請求到服務器 瀏覽器呈現來自緩存的頁面
no-cache/no-store 瀏覽器重新發送請求到服務器 瀏覽器重新發送請求到服務器 瀏覽器重新發送請求到服務器 瀏覽器重新發送請求到服務器
must-revalidation/proxy-revalidation 瀏覽器重新發送請求到服務器 第一次,瀏覽器重新發送請求到服務器;此後,瀏覽器呈現來自緩存的頁面 瀏覽器重新發送請求到服務器 瀏覽器呈現來自緩存的頁面
max-age=xxx (xxx is numeric) 在 xxx 秒後,瀏覽器重新發送請求到服務器 在 xxx 秒後,瀏覽器重新發送請求到服務器 瀏覽器重新發送請求到服務器 在 xxx 秒後,瀏覽器重新發送請求到服務器

瀏覽器行爲影響

在先前有效訪問後,在以後對同一URI資源的請求中,瀏覽器只進行兩種動作:

(1)直接在緩存中去獲取內容。如果先前有效訪問的響應頭包含 Expires,max-age的話,'打開新窗口' '輸入URI回車' '前一頁' '後一頁'這些瀏覽器行爲不會使瀏覽器在Expires,max-age設置的有效期時間內去訪問服務器,而是在緩存中去獲取內容,但是' 刷新' 或 '重載'例外。

(2)訪問服務器,根據服務器響應來獲取內容。這種情況發生在設置no-cache等頭標要求不緩存,或者是設置了 Expires,max-age但瀏覽器行爲是 ' 刷新' 或 '重載'時候。'Last-Modified' 'ETag' 'must-revalidate' 等有些特殊,不直接受瀏覽器行爲影響,它們也是訪問服務器後,再由服務器判斷是發送新的資源,還是發送一個304 Not Modfied讓瀏覽器使用緩存中的資源。

<?php   
/********************************
 * 客戶端緩存控制函數
 * $type  緩存類型
 * $interval  客戶端緩存過期時間
 * $mktime  設置Last-Modified
 * $etag  設置ETag標誌
 ******************************/  
function  http_cache_control( $type = 'nocache' , $interval =0, $mktime = '' , $etag = '' ){     
   if ( $type == 'nocache' )
   {     
       header('Expires: -1' );  //設置 -1爲立刻過期    
       header('Pragma: no-cache' );     
       header('Cache-Control: no-store, no-cache, must-revalidate, post-check=0, pre-check=0' );     
   }
   else 
   {   //檢查  ETag: 值   $_SERVER [ 'HTTP_IF_NONE_MATCH' ]
       if (isset( $_SERVER [ 'HTTP_IF_NONE_MATCH' ]) &&  $etag  &&  $_SERVER [ 'HTTP_IF_NONE_MATCH' ] ==  $etag )
       {     
            header('HTTP/1.1 304 Not Modfied' );     
       }//檢查  Last-Modified: 值   $_SERVER [ 'HTTP_IF_MODIFIED_SINCE' ]
       elseif (isset( $_SERVER [ 'HTTP_IF_MODIFIED_SINCE' ]) &&  $mktime  &&  $_SERVER [ 'HTTP_IF_MODIFIED_SINCE' ] ==  gmdate ( 'r' , $mktime ). ' GMT' )
       {     
           header('HTTP/1.1 304 Not Modfied' );     
       }
       else 
       {     //根據修改時間加過期時間,算出過期時間點
            if ( $mktime )
            {     
               $gmtime  =  gmdate ( 'r' , $mktime + $interval ). ' GMT' ;     
               header('Expires: ' . $gmtime );     
            }     
            if ( $type == 'public' )//設置緩存類型爲public
            {     
               header('Cache-Control: public,max-age=' . $interval );     
            }
            elseif ( $type == 'private' )//設置緩存類型爲 private
            {     
           	   header('Cache-Control: private,max-age=' . $interval . ',s-maxage=0' );     
            }elseif ( $type == 'none' )
            {     
               header('Cache-Control: must-revalidate,proxy-revalidate' );     
            }     
        }     
        $mktime && header( 'Last-Modified: ' . gmdate ( 'r' , $mktime ) . ' GMT' );     
        $etag   &&  header( 'ETag: ' . $etag );     
   }     
}     
?>    


 

發佈了23 篇原創文章 · 獲贊 1 · 訪問量 8萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章