4

我通过 PHP 脚本从 URL 向我的用户提供下载。使用时,readfile()我获得了我的连接可以支持的最大下载速度(大约 2.5MB/s),但是当我使用fopen, fread, fclose路由时,下载速度非常非常慢(大约 1-2KB/s)。

这是我的代码:

header('Content-Description: File Transfer');
header('Content-Type: application/octet-stream');
header('Content-Disposition: attachment; filename=' . $filename);
header('Content-Transfer-Encoding: binary');
header('Expires: 0');
header('Cache-Control: must-revalidate');
header('Pragma: public');
header('Content-Length: ' . $filesize);
ob_clean();
flush();

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

while(!feof($file)) {
    echo fread($file, 2014);
}

并且 readfile 代码很简单readfile($link);

我不能只使用这个readfile()功能有两个原因,一个是我想限制用户的下载速度(我可以fread通过只读取这么多的数据来做到这一点),我还想跟踪用户正在下载多少(我可以做到这一点,readfile()但它不计算部分下载)。

有谁知道为什么会发生这种情况或我该如何解决?据我所知readfile(),这只是一个包装,fopen, fread and fclose所以我不知道出了什么问题。

编辑:为此最终选择了 cURL。

$curl = curl_init();
$options = array(
    CURLOPT_URL => $rdLink,
    CURLOPT_FAILONERROR => true,
    CURLOPT_BINARYTRANSFER => true,
    CURLOPT_RETURNTRANSFER => true,
    CURLOPT_WRITEFUNCTION => 'readResponse'
);
curl_setopt_array($curl, $options);

if(!curl_exec($curl)) {
    header('Location: http://whatever.com');
    exit;
}
curl_close($curl);

function readResponse($ch, $data) {
    $length = mb_strlen($data, '8bit');

    echo $data;

    return $length;
}
4

3 回答 3

3

使用stream_context_create()stream_get_contents()

$context = stream_context_create();
$file = fopen($url, 'rb', FALSE, $context);
while(!feof($file))
{
    //usleep(1000000);
    echo stream_get_contents($file, 2014);
}

您还可以尝试使文件读取长度更大并放入 usleep() 以减慢执行速度。无论如何,对于最新版本的 PHP,似乎建议使用流函数而不是 fread。您可能还想在 fread() 或 stream_get_contents() 前面添加一个 @ 以抑制任何错误,至少在生产中是这样。没有它,还有一点点小事故,你就有了一个损坏的文件。

于 2013-02-03T08:34:11.920 回答
0

它可能在 PHP 或 Apache 的某个地方进行缓冲(甚至可能是速率限制)。尝试改变:

while(!feof($file)) {
   echo fread($file, 2014);
}

至:

while(!feof($file)) {
   $s=fread($file, 2014);
   if($s===false)break;  //Crude
   echo $s;
   @ob_flush();@flush();
}

@前缀是因为他们可能会抱怨空缓冲区。)

正如 mr.freshwater 所说,你应该对你的fread电话进行错误检查,所以我在上面添加了一些基本的东西。

于 2013-02-03T08:51:39.070 回答
0

原因是 2014 年。操作系统获得 4096 字节数据部分的速度比其他操作系统快。但是如果你写的是 2014 年,那么操作系统会尝试从文件中读取一个字节的数据来计算它。这就是为什么需要这么长时间。将 2014 更改为 4096

于 2017-07-20T12:59:20.597 回答