我认为我的问题看起来很随意,但随着它变得有趣(至少对我而言:),请耐心等待。)
考虑一个PHP页面,其目的是从文件系统读取请求的文件并将其作为响应回显。现在的问题是如何为此页面启用缓存?需要指出的是,文件可能非常庞大,启用缓存是为了避免客户端一次又一次地下载相同的内容。
理想的策略是使用“If-None-Match”请求头和“ETag”响应头来实现反向代理缓存系统。虽然我知道这一点,但我不确定这是否可行,或者我应该将什么作为响应返回以实现此技术!
答案 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
来确定内容是否同时发生了变化。