Apache/PHP大文件下载(> 2Gb)失败

Ric*_*ick 8 php apache

我正在使用PHP脚本来控制对下载文件的访问.这适用于2Gb以下的任何内容,但对于较大的文件则无效.

  • Apache和PHP都是64位
  • 如果直接访问,Apache 允许下载文件(我不允许)

PHP的内核(忽略访问控制):

if (ob_get_level())  ob_end_clean();

error_log('FILETEST: '.$path.' : '.filesize($path));
header('Content-Description: File Transfer');
header('Content-Type: application/octet-stream');
header('Content-Disposition: attachment; filename='.basename($path));
header('Expires: 0');
header('Cache-Control: must-revalidate');
header('Pragma: public');
header('Content-Length: ' . filesize($path));
readfile($path);
exit;
Run Code Online (Sandbox Code Playgroud)

错误日志显示文件大小正常

[Tue Apr 08 11:01:16 2014] [error] [client *.*.*.*] FILETEST: /downloads/file.name : 2251373807, referer: http://myurl/files/
Run Code Online (Sandbox Code Playgroud)

但访问日志的大小为负:

 *.*.*.* - - [08/Apr/2014:11:01:16 +0100] "GET /files/file.name HTTP/1.1" 200 -2043593489 "http://myurl/files/" "Mozilla/5.0 (Windows NT 6.1; WOW64; rv:24.0) Gecko/20100101 Firefox/24.0"
Run Code Online (Sandbox Code Playgroud)

因此浏览器拒绝下载该文件.事实上,使用wget,它不会发送任何东西:

$ wget -S -O - http://myurl/files/file.name
--2014-04-08 11:33:38--  http://myurl/files/file.name
HTTP request sent, awaiting response... No data received.
Retrying.
Run Code Online (Sandbox Code Playgroud)

Dan*_* W. 8

尝试以块的形式读取文件并将它们暴露给浏览器,而不是用2GB填充本地内存并立即刷新所有文件.

替换readfile($path);为:

@ob_end_flush();
flush();

$fileDescriptor = fopen($file, 'rb');

while ($chunk = fread($fileDescriptor, 8192)) {
    echo $chunk;
    @ob_end_flush();
    flush();
}

fclose($fileDescriptor);
exit;
Run Code Online (Sandbox Code Playgroud)

在某些情况下,8192字节是关键点,参考php.net/fread.

添加一些microtime变量(并与文件描述符的指针位置进行比较)也可以控制下载的最大速度.

*(刷新输出缓冲区也略微取决于Web服务器,使用这些命令确保它至少尝试尽可能地刷新.)

  • 使用这个而不是读取文件的下载开始下载要快得多..正在努力这一段时间:我欠你一杯啤酒......! (2认同)