14

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

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

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

4

1 回答 1

25

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

相反,请查看用于 nginx的X-accel 、用于 Lighttpd的 X-Sendfile或用于 Apache的mod_xsendfile 。

初始请求由 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用来判断内容是否同时发生了变化。

于 2012-05-15T07:40:08.060 回答