从URL下载文件到服务器

这个看起来很简单,事实也的确如此。你所要做的就是下载一个文件到你的服务器:

file_put_contents("Tmpfile.zip", file_get_contents("http://someurl/file.zip"));

只有一个问题。如果你有一个大文件,比如100mb。然后,您将耗尽内存,无法下载文件。

我想要的是一种将文件写入磁盘的方法,因为我正在下载它。这样,我就可以下载更大的文件,而不会遇到内存问题。

675192 次浏览

自PHP 5.1.0起,file_put_contents()支持通过传递一个流句柄作为$data参数来逐条写入:

file_put_contents("Tmpfile.zip", fopen("http://someurl/file.zip", 'r'));

摘自手册:

如果< em > < / em >数据[这是第二个参数]是一个流资源,该流的剩余缓冲区将被复制到指定的文件。这与使用类似 stream_copy_to_stream() . < / p >

(谢谢Hakre。)

尝试使用cURL

set_time_limit(0); // unlimited max execution time
$options = array(
CURLOPT_FILE    => '/path/to/download/the/file/to.zip',
CURLOPT_TIMEOUT =>  28800, // set this to 8 hours so we dont timeout on big files
CURLOPT_URL     => 'http://remoteserver.com/path/to/big/file.zip',
);


$ch = curl_init();
curl_setopt_array($ch, $options);
curl_exec($ch);
curl_close($ch);

我不确定,但我相信CURLOPT_FILE选项在它提取数据时写入。没有缓冲。

private function downloadFile($url, $path)
{
$newfname = $path;
$file = fopen ($url, 'rb');
if ($file) {
$newf = fopen ($newfname, 'wb');
if ($newf) {
while(!feof($file)) {
fwrite($newf, fread($file, 1024 * 8), 1024 * 8);
}
}
}
if ($file) {
fclose($file);
}
if ($newf) {
fclose($newf);
}
}
  1. 创建一个名为“下载”的文件夹;在目标服务器中
  2. 将[此代码]保存到.php文件中并在目标服务器中运行

下载:

<html>
<form method="post">
<input name="url" size="50" />
<input name="submit" type="submit" />
</form>
<?php
// maximum execution time in seconds
set_time_limit (24 * 60 * 60);


if (!isset($_POST['submit'])) die();


// folder to save downloaded files to. must end with slash
$destination_folder = 'downloads/';


$url = $_POST['url'];
$newfname = $destination_folder . basename($url);


$file = fopen ($url, "rb");
if ($file) {
$newf = fopen ($newfname, "wb");


if ($newf)
while(!feof($file)) {
fwrite($newf, fread($file, 1024 * 8 ), 1024 * 8 );
}
}


if ($file) {
fclose($file);
}


if ($newf) {
fclose($newf);
}
?>
</html>
set_time_limit(0);
$file = file_get_contents('path of your file');
file_put_contents('file.ext', $file);

我用这个来下载文件

function cURLcheckBasicFunctions()
{
if( !function_exists("curl_init") &&
!function_exists("curl_setopt") &&
!function_exists("curl_exec") &&
!function_exists("curl_close") ) return false;
else return true;
}


/*
* Returns string status information.
* Can be changed to int or bool return types.
*/
function cURLdownload($url, $file)
{
if( !cURLcheckBasicFunctions() ) return "UNAVAILABLE: cURL Basic Functions";
$ch = curl_init();
if($ch)
{


$fp = fopen($file, "w");
if($fp)
{
if( !curl_setopt($ch, CURLOPT_URL, $url) )
{
fclose($fp); // to match fopen()
curl_close($ch); // to match curl_init()
return "FAIL: curl_setopt(CURLOPT_URL)";
}
if ((!ini_get('open_basedir') && !ini_get('safe_mode')) || $redirects < 1) {
curl_setopt($ch, CURLOPT_USERAGENT, '"Mozilla/5.0 (X11; U; Linux i686; en-US; rv:1.8.1.11) Gecko/20071204 Ubuntu/7.10 (gutsy) Firefox/2.0.0.11');
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
//curl_setopt($ch, CURLOPT_REFERER, 'http://domain.com/');
if( !curl_setopt($ch, CURLOPT_HEADER, $curlopt_header)) return "FAIL: curl_setopt(CURLOPT_HEADER)";
if( !curl_setopt($ch, CURLOPT_FOLLOWLOCATION, $redirects > 0)) return "FAIL: curl_setopt(CURLOPT_FOLLOWLOCATION)";
if( !curl_setopt($ch, CURLOPT_FILE, $fp) ) return "FAIL: curl_setopt(CURLOPT_FILE)";
if( !curl_setopt($ch, CURLOPT_MAXREDIRS, $redirects) ) return "FAIL: curl_setopt(CURLOPT_MAXREDIRS)";


return curl_exec($ch);
} else {
curl_setopt($ch, CURLOPT_USERAGENT, '"Mozilla/5.0 (X11; U; Linux i686; en-US; rv:1.8.1.11) Gecko/20071204 Ubuntu/7.10 (gutsy) Firefox/2.0.0.11');
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
//curl_setopt($ch, CURLOPT_REFERER, 'http://domain.com/');
if( !curl_setopt($ch, CURLOPT_FOLLOWLOCATION, false)) return "FAIL: curl_setopt(CURLOPT_FOLLOWLOCATION)";
if( !curl_setopt($ch, CURLOPT_FILE, $fp) ) return "FAIL: curl_setopt(CURLOPT_FILE)";
if( !curl_setopt($ch, CURLOPT_HEADER, true)) return "FAIL: curl_setopt(CURLOPT_HEADER)";
if( !curl_setopt($ch, CURLOPT_RETURNTRANSFER, true)) return "FAIL: curl_setopt(CURLOPT_RETURNTRANSFER)";
if( !curl_setopt($ch, CURLOPT_FORBID_REUSE, false)) return "FAIL: curl_setopt(CURLOPT_FORBID_REUSE)";
curl_setopt($ch, CURLOPT_USERAGENT, '"Mozilla/5.0 (X11; U; Linux i686; en-US; rv:1.8.1.11) Gecko/20071204 Ubuntu/7.10 (gutsy) Firefox/2.0.0.11');
}
// if( !curl_setopt($ch, CURLOPT_FOLLOWLOCATION, true) ) return "FAIL: curl_setopt(CURLOPT_FOLLOWLOCATION)";
// if( !curl_setopt($ch, CURLOPT_FILE, $fp) ) return "FAIL: curl_setopt(CURLOPT_FILE)";
// if( !curl_setopt($ch, CURLOPT_HEADER, 0) ) return "FAIL: curl_setopt(CURLOPT_HEADER)";
if( !curl_exec($ch) ) return "FAIL: curl_exec()";
curl_close($ch);
fclose($fp);
return "SUCCESS: $file [$url]";
}
else return "FAIL: fopen()";
}
else return "FAIL: curl_init()";
}

A PHP 4 &5解决方案:

readfile ()本身不会出现任何内存问题,即使在发送大文件时也是如此。 如果fopen包装器已启用,则URL可以用作此函数的文件名

http://php.net/manual/en/function.readfile.php

有3种方法:

  1. File_get_contents和file_put_contents
  2. 旋度
  3. 打开外部文件

你可以找到例子从这里

浪子的回答对我不起作用。我得到missing fopen in CURLOPT_FILE 更多的细节

这对我来说很有效,包括本地url:

function downloadUrlToFile($url, $outFileName)
{
if(is_file($url)) {
copy($url, $outFileName);
} else {
$options = array(
CURLOPT_FILE    => fopen($outFileName, 'w'),
CURLOPT_TIMEOUT =>  28800, // set this to 8 hours so we dont timeout on big files
CURLOPT_URL     => $url
);


$ch = curl_init();
curl_setopt_array($ch, $options);
curl_exec($ch);
$httpcode = curl_getinfo($ch, CURLINFO_HTTP_CODE);
curl_close($ch);
return $httpcode;
}
}


        

使用php copy()中的一个简单方法

copy($source_url, $local_path_with_file_name);

注意:如果目标文件已经存在,它将被覆盖

PHP copy()函数 . PHP copy(

备注:需要为目标文件夹设置权限777。 当你下载到你的本地机器时,使用这个方法

特殊提示:777是基于Unix系统的权限,对所有者、组和所有人都具有完全的读/写/执行权限。通常,我们将此权限授予在web服务器上不太需要对公共隐藏的资产。例如:images文件夹。

简单的解决方案:

<?php
exec('wget http://someurl/file.zip');

最好的解决方案

在系统&中安装aria2c;

 echo exec("aria2c \"$url\"")