我使用以下内容下载PHP文件:
ob_start();
$browser = id_browser();
header('Content-Type: '.(($browser=='IE' || $browser=='OPERA')?
'application/octetstream':'application/octet-stream'));
header('Expires: '.gmdate('D, d M Y H:i:s').' GMT');
header('Content-Transfer-Encoding: binary');
header('Content-Length: '.filesize(realpath($fullpath)));
//header("Content-Encoding: none");
if($browser == 'IE')
{
header('Content-Disposition: attachment; filename="'.$file.'"');
header('Cache-Control: must-revalidate, post-check=0, pre-check=0');
header('Pragma: public');
} else
{
header('Content-Disposition: attachment; filename="'.$file.'"');
header('Cache-Control: no-cache, must-revalidate');
header('Pragma: no-cache');
}
//@set_time_limit( 0 );
ReadFileChunked(utf8_decode($fullpath));
ob_end_flush();
ReadFileChunked的源代码是:
function ReadFileChunked($filename,$retbytes=true)
{
$chunksize = 1*(1024*1024);
$remainFileSize = filesize($filename);
if($remainFileSize < $chunksize)
$chunksize = $remainFileSize;
$buffer = '';
$cnt =0;
// $handle = fopen($filename, 'rb');
//echo $filename."<br>";
$handle = fopen($filename, 'rb');
if ($handle === false) {
//echo 1;
return false;
}
//echo 2;
while (!feof($handle))
{
//echo "current remain file size $remainFileSize<br>";
//echo "current chunksize $chunksize<br>";
$buffer = fread($handle, $chunksize);
echo $buffer;
sleep(1);
ob_flush();
flush();
if ($retbytes) {
$cnt += strlen($buffer);
}
$remainFileSize -= $chunksize;
if($remainFileSize == 0)
break;
if($remainFileSize < $chunksize)
{
$chunksize = $remainFileSize;
}
}
$status = fclose($handle);
if ($retbytes && $status) {
return $cnt; // return num. bytes delivered like readfile() does.
}
return $status;
}
问题是: 下载的文件将包含一些html标签,这些标签是php生成的html代码的内容。
下载文件大小小于4096字节的txt文件时会发生错误。
请帮我解决这个问题,非常感谢你!
储
答案 0 :(得分:1)
您是否尝试过使用fpassthru而非自定义功能。
答案 1 :(得分:0)
没有必要在那里使用$ chunksize的东西。 fread()
一旦到达文件末尾就会自动停止读取,即使$ chunksize通常会告诉它读取更多内容。同样,您应该在ob_flush()
之前调出flush()
和sleep(1)
来电。这样,您刚放入输出缓冲区的数据就可以被发送到网络服务器而不必等待一秒钟。
实际上,您可以使用以下内容替换整个函数:
function ReadFileChunk($filename, $retbytes = true) {
$fh = fopen($filename, 'rb');
if (!$fh) {
return(false);
}
while($buf = fread($fh, 4096)) {
echo $buf;
ob_flush();
flush();
sleep(1);
}
$status = fclose($fh);
return( $retbytes ? filesize($filename) : $status);
}
但是,当readfile()
已经存在时,为什么还要自己动手?它将处理打开文件的整个业务,并以不超过memory_limit
的正常大小的片段发送。