1


我正在测试此代码以下载 10 MB 块的大文件:

/**
 * Copy remote file over HTTP one small chunk at a time.
 *
 * @param $infile The full URL to the remote file
 * @param $outfile The path where to save the file
 */
function copyfile_chunked($infile, $outfile) {
    $chunksize = 10 * (1024 * 1024); // 10 Megs

    /**
     * parse_url breaks a part a URL into it's parts, i.e. host, path,
     * query string, etc.
     */
    $parts = parse_url($infile);
    $i_handle = fsockopen($parts['host'], 80, $errstr, $errcode, 5);
    $o_handle = fopen($outfile, 'wb');

    if ($i_handle == false || $o_handle == false) {
        return false;
    }

    if (!empty($parts['query'])) {
        $parts['path'] .= '?' . $parts['query'];
    }

    /**
     * Send the request to the server for the file
     */
    $request = "GET {$parts['path']} HTTP/1.1\r\n";
    $request .= "Host: {$parts['host']}\r\n";
    $request .= "User-Agent: Mozilla/5.0\r\n";
    $request .= "Keep-Alive: 115\r\n";
    $request .= "Connection: keep-alive\r\n\r\n";
    fwrite($i_handle, $request);

    /**
     * Now read the headers from the remote server. We'll need
     * to get the content length.
     */
    $headers = array();
    while(!feof($i_handle)) {
        $line = fgets($i_handle);
        if ($line == "\r\n") break;
        $headers[] = $line;
    }

    /**
     * Look for the Content-Length header, and get the size
     * of the remote file.
     */
    $length = 0;
    foreach($headers as $header) {
        if (stripos($header, 'Content-Length:') === 0) {
            $length = (int)str_replace('Content-Length: ', '', $header);
            break;
        }
    }

    /**
     * Start reading in the remote file, and writing it to the
     * local file one chunk at a time.
     */
    $cnt = 0;
    while(!feof($i_handle)) {
        $buf = '';
        $buf = fread($i_handle, $chunksize);
        $bytes = fwrite($o_handle, $buf);
        if ($bytes == false) {
            return false;
        }
        $cnt += $bytes;

        /**
         * We're done reading when we've reached the conent length
         */
        if ($cnt >= $length) break;
    }

    fclose($i_handle);
    fclose($o_handle);
    return $cnt;
}

我首先在一个小图像上测试这个代码。该图像已下载到我的帐户,但格式已损坏:所有字节似乎都是正确的,除了“0D”字节已从下载的图像中删除,这使其无法使用。
为什么会发生这种情况,我该如何克服?
谢谢!

4

1 回答 1

1

大家好,感谢您的帮助。
现在问题已经解决,罪魁祸首已经确定。
我一直在阅读一些书籍并发现:
ftp_get() 将远程服务器上的文件复制到您的计算机。FTP_ASCII 参数像传输 ASCII 文本一样传输文件。在此选项下,当您从一个操作系统移动到另一个操作系统时,会自动转换换行符结尾。另一个选项是 FTP_BINARY,它用于非纯文本文件,因此不会发生换行转换。
我的问题中提供的代码工作正常,它可以正确下载图像。
当我检查图像时,我正在使用由托管服务提供商提供的 php 编写的文件管理器将其下载到我的计算机上。显然,他们不太擅长 PHP,因为他们使用上面提到的 FTP_ASCII 参数来传输二进制文件。因此图像被破坏了。
当我直接从 FTP 帐户下载图像时,证明图像与原始图像相同。
所以,最终,问题出在 PHP 代码上,而不是我编译的代码上。

于 2012-11-09T02:55:15.643 回答