PHP客户端缓存控制

-by 落杉

 

客户端缓存控制通常也会被叫做条件GET请求。

五种常用于控制客户端缓存的头标

  • Last-Modified (最后修改时间)
  • ETag (实体标签)
  • Expires (有效指示)
  • Pragma (编译指示)
  • Cache-Control (缓存控制)

1.Last-Modified

这个头标是一个响应头标,表示客户端(通常指浏览器)所请求资源在服务器端的最后修改时间,通常情况下客户端在接受这个头标后,在以后对这个资源的 请求会附带一个'If-Modified-Since'请求头标,而这个头标是想告诉服务器上次客户端所请求资源的最后修改时间,对于一些图 像,css,js等静态文件资源,配置好了的apache服务器会理解这些If-Modified-Since请求头标,将头标里的时间和文件的最后修改 时间进行比较并作出响应,如果二者相等则发送一个304 Not Modfied来告诉客户端所请求资源并未修改让客户端放心使用缓存中的资源,否则的话会重新发送一个新的资源和新的Last-Modified的头标。 但是对于一个动态的PHP脚本,我们即使在脚本加入了header('Last Modified: '.$time)来发送一个Last Modified响应头标,当客户端附带'If-Modified-Since'在次请求时apache服务器不会进行处理,这需要我们自己 用$_SERVER['HTTP_IF_MODIFIED_SINCE']来获取'If-Modified-Since'的值自己来进行判断处理。

2.ETag

ETag(Entity Tag)和Last-Modified类似,也是WEB服务器和客户端用于确认缓存组件的有效性的一种机制,apache 1.3和2.0的ETag格式是inode-size-timestamp,因此当资源被修改,其ETag也发生改变,ETag相对Last- Modified更精确,Last-Modified只能精确的s级别,但是ETag在多服务器可能造成混乱,所以用还是不用还得看实际情况,其相对应的 后续请求头标为If-None-Match。

3.Expires

Expires表明缓存何时因该过期(GMT时间),属于HTTP 1.0 标准,通常是用来对Cache-Control的max-age的一个补充,来兼容HTTP 1.0,不赞成单独使用Expires,因为客户端时间容易发生偏差。

4.Pragma

HTTP 1.0 标准,通常是在不缓存时使用,Pragma: no-cache。

5.Cache-Control

Cache-Control常用指令(暂时我还没找到完整版本的)

public 可以在任何地方缓存
private 只能被浏览器缓存
no-cache 不能在任何地方缓存
must-revalidate 缓存必须检查跟新版本
proxy-revalidate 代理缓存必须检查跟新版本
max-age 内容能够被缓存的时间
s-maxage 覆盖共享缓存的max-age设置

浏览器行为影响

在先前有效访问后,在以后对同一URI资源的请求中,浏览器只进行两种动作:(1)直接在缓存中去获取内容。如果先前有效访问的响应头包含 Expires,max-age的话,'打开新窗口' '输入URI回车' '前一页' '后一页'这些浏览器行为不会使浏览器在Expires,max-age设置的有效期时间内去访问服务器,而是在缓存中去获取内容,但是' 刷新' 或 '重载'例外。(2)访问服务器,根据服务器响应来获取内容。这种情况发生在设置no-cache等头标要求不缓存,或者是设置了 Expires,max-age但浏览器行为是 ' 刷新' 或 '重载'时候。'Last-Modified' 'ETag' 'must-revalidate' 等有些特殊,不直接受浏览器行为影响,它们也是访问服务器后,再由服务器判断是发送新的资源,还是发送一个304 Not Modfied让浏览器使用缓存中的资源。

PHP代码
  1. <?php  
  2. function  http_cache_control( $type = 'nocache' , $interval =0, $mktime = '' , $etag = '' ){  
  3.     if ( $type == 'nocache' ){  
  4.         header('Expires: -1' );  
  5.         header('Pragma: no-cache' );  
  6.         header('Cache-Control: no-store, no-cache, must-revalidate, post-check=0, pre-check=0' );  
  7.     }else {  
  8.         if (isset( $_SERVER [ 'HTTP_IF_NONE_MATCH' ]) &&  $etag  &&  $_SERVER ['HTTP_IF_NONE_MATCH' ] ==  $etag ){  
  9.             header('HTTP/1.1 304 Not Modfied' );  
  10.         }elseif (isset( $_SERVER [ 'HTTP_IF_MODIFIED_SINCE' ]) &&  $mktime  &&  $_SERVER ['HTTP_IF_MODIFIED_SINCE' ] ==  gmdate ( 'r' , $mktime ). ' GMT' ){  
  11.             header('HTTP/1.1 304 Not Modfied' );  
  12.         }else {  
  13.             if ( $mktime ){  
  14.                 $gmtime  =  gmdate ( 'r' , $mktime + $interval ). ' GMT' ;  
  15.                 header('Expires: ' . $gmtime );  
  16.             }  
  17.             if ( $type == 'public' ){  
  18.                 header('Cache-Control: public,max-age=' . $interval );  
  19.             }elseif ( $type == 'private' ){  
  20.                 header('Cache-Control: private,max-age=' . $interval . ',s-maxage=0' );  
  21.             }elseif ( $type == 'none' ){  
  22.                 header('must-revalidate,proxy-revalidate' );  
  23.             }  
  24.         }  
  25.         $mktime && header( 'Last-Modified: ' . gmdate ( 'r' , $mktime ) . ' GMT' );  
  26.         $etag    &&  header( 'ETag: ' . $etag );  
  27.     }  
  28. }  
  29. ?>  

转载于:https://my.oschina.net/bugyang/blog/121836

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值