14

现在,我正在尝试从不允许我启用mod_gzipmod_deflate. 于是我写了一个PHP小脚本,用GZIP压缩并返回给用户。

示例代码:

$filename = "style.css";

if (!file_exists($filename) || !($info = stat($filename))) {
  header("HTTP/1.1 404 Not Found");
  die();
}

header("Date: ".gmdate("D, j M Y H:i:s e", time()));
header("Cache-Control: max-age=2592000");
header("Last-Modified: ".gmdate("D, j M Y H:i:s e", $info['mtime']));
header("Etag: ".sprintf("\"%x-%x-%x\"", $info['ino'], $info['size'], $info['mtime']));
header("Accept-Ranges: bytes");
header("Cache-Control: Expires ".gmdate("D, j M Y H:i:s e", $info['mtime']+2592000));
header("Content-Type: text/html");

ob_start("ob_gzhandler");
echo file_get_contents($filename);
ob_end_flush();

我现在有两个问题。首先是,我无法确定压缩文件的最终大小以通知浏览器内容长度。通常,我会包括这一行:

header("Content-Length: ".$info["size"]);

但是,如果我这样做了,浏览器会在尝试等待更多数据时挂起。有没有办法计算总大小?或者我应该忽略这个头指令。

另一个问题是,每当我在 Firefox 中查看这个 PHP 文件时,它都会尝试让我下载结果。在 Chrome 中,它只是像我期望的那样显示它。有什么建议么?

编辑:感谢 SoapBox,我用以下代码替换了代码的结尾:

header("Content-Encoding: gzip");
$compressed = gzencode(file_get_contents($filename), 5);
header("Content-Length: ".strlen($compressed));
die($compressed);

这对内容长度非常有用!但我仍然让 Firefox 下载文件而不是显示它。:(

再次编辑:这是修改后的代码结尾代码,由 Cletus 提供。

// Start buffered output
ob_start();
// Check for gzip capability
if (stripos($_SERVER['HTTP_ACCEPT_ENCODING'], "gzip") !== false) {
  ob_start("ob_gzhandler");
  echo file_get_contents($filename);
  ob_end_flush();
} else
  echo file_get_contents($filename);

// Write the content length
header('Content-Length: '.ob_get_length());
ob_end_flush();

我将开始一个新问题来弄清楚为什么 Firefox 一直尝试下载该文件。

4

3 回答 3

19

这里的问题是,要知道内容长度,您需要知道客户端是否支持 gzip 编码,并且您已经通过使用 ob_gzhandler 委派了该决定。从HTTP 标头

ob_start();
ob_start('ob_gzhandler');

  ... output the page content...

ob_end_flush();  // The ob_gzhandler one

header('Content-Length: '.ob_get_length());

ob_end_flush();  // The main one

完整版:

$filename = "style.css";

if (!file_exists($filename) || !($info = stat($filename))) {
  header("HTTP/1.1 404 Not Found");
  die();
}

header("Date: ".gmdate("D, j M Y H:i:s e", time()));
header("Cache-Control: max-age=2592000");
header("Last-Modified: ".gmdate("D, j M Y H:i:s e", $info['mtime']));
header("ETag: ".sprintf("\"%x-%x-%x\"", $info['ino'], $info['size'], $info['mtime']));
header("Accept-Ranges: bytes");
header("Expires: ".gmdate("D, j M Y H:i:s e", $info['mtime']+2592000));
header("Content-Type: text/css"); // note: this was text/html for some reason?

ob_start();
ob_start("ob_gzhandler");
echo file_get_contents($filename);
ob_end_flush();
header('Content-Length: '.ob_get_length());
ob_end_flush();

这比自己处理 gzip 编码问题要好得多

于 2009-05-03T01:28:04.127 回答
1

您需要首先执行整个 gzip 并测量结果(将内容保存在内存中,或者在压缩时将它们写入磁盘,然后统计 gzip 压缩文件),然后写入 Content-Length 标头,然后发送文件内容。

或者使用分块传输编码

于 2009-05-03T00:12:24.963 回答
1

为了解决您的Firefox问题,我认为您需要包含header( "Content-Encoding: gzip" );以便浏览器知道解压缩内容。

至于内容长度,你可以试着把这个值去掉,或者想办法使用“Transfer-Encoding:chunked”(你不能只发送这个header,你需要专门为它格式化数据)。可以ob_end_flush自动启用分块。

我建议您获取 wireshark 并捕获您的 php 脚本正在发送的内容,并将其与行为正常的服务器进行比较,以查看缺少哪些标头等。

于 2009-05-03T00:15:48.123 回答