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