在PHP动态创建HTTP响应时缓存它们

时间:2012-05-15 07:30:45

标签: php caching http-headers

我认为我的问题看起来很随意,但随着它变得有趣(至少对我而言:),请耐心等待。)

考虑一个PHP页面,其目的是从文件系统读取请求的文件并将其作为响应回显。现在的问题是如何为此页面启用缓存?需要指出的是,文件可能非常庞大,启用缓存是为了避免客户端一次又一次地下载相同的内容。

理想的策略是使用“If-None-Match”请求头和“ETag”响应头来实现反向代理缓存系统。虽然我知道这一点,但我不确定这是否可行,或者我应该将什么作为响应返回以实现此技术!

1 个答案:

答案 0 :(得分:25)

使用PHP提供大量或许多辅助文件并不完全是为它做的。

相反,请查看X-accel获取nginx,X-Sendfile获取Lighttpd或mod_xsendfile获取Apache。

初始请求由PHP处理,但是一旦确定了下载文件,它就会设置一些标题来指示服务器应该处理文件发送,之后PHP进程被释放以提供其他服务。

然后,您可以使用Web服务器为您配置缓存。

静态生成内容

如果您的内容是从PHP生成的并且创建起来特别昂贵,您可以将输出写入本地文件并再次应用上述方法。

如果您无法写入本地文件或不想写,则可以使用HTTP响应标头来控制缓存:

Expires: <absolute date in the future>
Cache-Control: public, max-age=<relative time in seconds since request>

这将导致客户端缓存页面内容,直到它过期或用户强制页面重新加载(例如按F5)。

动态生成内容

对于动态内容,您希望浏览器每次都ping您,但只有在有新内容时才发送页面内容。您可以通过设置一些其他响应标头来完成此操作:

ETag: <hash of the contents>
Last-Modified: <absolute date of last contents change>

当浏览器再次ping您的脚本时,他们将分别添加以下请求标头:

If-None-Match: <hash of the contents that you sent last time>
If-Modified-Since: <absolute date of last contents change>

ETag主要用于减少网络流量,因为在某些情况下,要了解内容哈希,首先必须计算它。

如果您有本地文件缓存(文件具有修改日期),则Last-Modified最容易应用。一个简单的条件使它工作:

if (!file_exists('cache.txt') || 
    filemtime('cache.txt') > strtotime($_SERVER['HTTP_IF_MODIFIED_SINCE'])) {
    // update cache file and send back contents as usual (+ cache headers)
} else {
    header('HTTP/1.0 304 Not modified');
}

如果你不能进行文件缓存,你仍然可以使用ETag来确定内容是否同时发生了变化。