【问题标题】:How to determine the Content-Length of a gzipped file?如何确定压缩文件的内容长度?
【发布时间】:2010-10-23 09:23:29
【问题描述】:

现在我正在尝试从不允许我启用 mod_gzipmod_deflate 的服务器提供 CSS 和 JS 文件。于是写了一个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 一直尝试下载文件。

【问题讨论】:

  • 答案中提到的一些标题有一些错误;双 Cache-Control,带有 Cache-Control: Expires.... 的行应该是 Expires: ... 而 Etag 应该是 ETag。

标签: php http-headers gzip header


【解决方案1】:

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

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 编码问题好多了

【讨论】:

    【解决方案2】:

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

    或使用chunked transfer encoding

    【讨论】:

      【解决方案3】:

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

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

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

      【讨论】:

      • 如果客户端支持,其实 ob_start("ob_gzhandler") 就是这么做的。
      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2011-04-18
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多