强制下载PDF文件,损坏文件

时间:2013-04-08 21:43:54

标签: php file corrupt force-download

我有一个问题,已经多次上升,但我似乎无法找到我的解决方案!我正在尝试将pdf文件发送到客户端而不在浏览器中打开,文件下载但是当我打开它并且从原始文件中丢失了相当多的字节时它已损坏。我已经尝试了几种这样的方法来下载文件,但我只会向您展示我使用过的最新版本并希望得到一些反馈。

我还在文本编辑器中打开了下载的PDF,并且我可以看到它顶部没有php错误!

我也知道readfile()要快得多,但出于测试目的,我迫不及待地想要做任何事情,所以我使用了while(!feof())方法!

无论如何漫无边际,继承代码(取自why my downloaded file is alwayes damaged or corrupted?):

$file     = __DIR__ . '/reports/somepdf.pdf';
$basename = basename($file);
$length   = sprintf("%u", filesize($file));

header('Content-Description: File Transfer');
header('Content-Type: application/octet-stream');
header('Content-Disposition: attachment; filename="' . $basename . '"');
header('Content-Transfer-Encoding: binary');
header('Connection: Keep-Alive');
header('Expires: 0');
header('Cache-Control: must-revalidate, post-check=0, pre-check=0');
header('Pragma: public');
header('Content-Length: ' . $length);

ob_clean();
set_time_limit(0);
readfile($file);

另外需要注意的是文件大小的差异:

Original: 351,873 bytes
Downloaded: 329,163 bytes

3 个答案:

答案 0 :(得分:8)

确保您没有运行任何压缩输出缓冲处理程序,例如ob_gzhandler。我有类似的情况,我必须禁用输出缓冲才能正常工作

答案 1 :(得分:7)

您正在使用输出缓冲区上的ob_gzhandler

它通过gzencoding输出块来工作。然后输出是编码块的流。

每个块需要获取一些字节才能进行编码,因此输出稍微缓冲,直到有足够的字节可用。

但是,在脚本结束时,您将丢弃剩余的缓冲区而不是刷新它。

使用ob_end_flush()代替ob_clean(),文件完全通过而不会损坏。

你要使用ob_gzhandler的传输编码,文件上传没有任何问题,如果你没有破坏输出缓冲区就可以完成它的工作。

如果任何其他输出缓冲工作,那么这也是一样的 启用。

示例代码:

$file     = __DIR__ . '/somepdf.pdf';
$basename = basename($file);
$length   = sprintf("%u", filesize($file));

header('Content-Description: File Transfer');
header('Content-Type: application/octet-stream');
header('Content-Disposition: attachment; filename="' . $basename . '"');
header('Content-Transfer-Encoding: binary');
header('Connection: Keep-Alive');
header('Expires: 0');
header('Cache-Control: must-revalidate, post-check=0, pre-check=0');
header('Pragma: public');
header('Content-Length: ' . $length);

ob_end_flush();   // <--- instead of ob_clean()

set_time_limit(0);
readfile($file);

return;

(仅供参考:实际上即使ob_end_flush();也没有必要,重要的部分不是仅仅在输出缓冲区完成它之前启动输出缓冲区)

答案 2 :(得分:-1)

在找到问题的解决方案之前,我使用内容处理来推动PDF下载两天。我的PDF文件尺寸也较小且已损坏 - 但是,我可以在Windows预览版中打开它们 - 而不是Adobe。经过多次故障排除后,我发现Adobe期望文件的前1024个字节中包含%PDF。在创建标题之前,我在我的php代码中进行了所有文件类型检查。我在标题和我的PDF文件修复之前取出了大部分代码。

您可能没有像我一样设置它,但可能是同样的问题:

http://helpx.adobe.com/acrobat/kb/pdf-error-1015-11001-update.html