PHP完全自学手册(珍藏版) 中文pdf扫描版下载
最近导出订单信息的时候出现一个php内存溢出的问题,原因是文件在下载的时候读取生成的临时文件过大,php内存无法容纳,开始是想更改php内存限制,但不是长久之计,于是就想到了把文件分次读取,并下载的方法。
以下是原来的源代码:
<?php $sourceFile = "1.tmp"; //要下载的临时文件名,可以用几百M的文件来测试 $outFile = "用户订单.xls"; //下载保存到客户端的文件名 $file_extension = strtolower(substr(strrchr($sourceFile, "."), 1)); //获取文件扩展名 //echo $sourceFile; if (!ereg("[tmp|txt|rar|pdf|doc]", $file_extension))exit ("非法资源下载"); //检测文件是否存在 if (!is_file($sourceFile)) { die("<b>404 File not found!</b>"); } $len = filesize($sourceFile); //获取文件大小 $filename = basename($sourceFile); //获取文件名字 $outFile_extension = strtolower(substr(strrchr($outFile, "."), 1)); //获取文件扩展名 //根据扩展名 指出输出浏览器格式 switch ($outFile_extension) { case "exe" : $ctype = "application/octet-stream"; break; case "zip" : $ctype = "application/zip"; break; case "mp3" : $ctype = "audio/mpeg"; break; case "mpg" : $ctype = "video/mpeg"; break; case "avi" : $ctype = "video/x-msvideo"; break; default : $ctype = "application/force-download"; } //Begin writing headers //header("Cache-Control:"); header("Cache-Control: public"); //设置输出浏览器格式 header("Content-Type: $ctype"); header("Content-Disposition: attachment; filename=" . $outFile); header("Accept-Ranges: bytes"); $size = filesize($sourceFile); //如果有$_SERVER['HTTP_RANGE']参数 if (isset ($_SERVER['HTTP_RANGE'])) { /*Range头域 Range头域可以请求实体的一个或者多个子范围。 例如, 表示头500个字节:bytes=0-499 表示第二个500字节:bytes=500-999 表示最后500个字节:bytes=-500 表示500字节以后的范围:bytes=500- 第一个和最后一个字节:bytes=0-0,-1 同时指定几个范围:bytes=500-600,601-999 但是服务器可以忽略此请求头,如果无条件GET包含Range请求头,响应会以状态码206(PartialContent)返回而不是以200 (OK)。 */ // 断点后再次连接 $_SERVER['HTTP_RANGE'] 的值 bytes=4390912- list ($a, $range) = explode("=", $_SERVER['HTTP_RANGE']); //if yes, download missing part //str_replace($range, "-", $range); //这句干什么的呢。。。。 $size2 = $size -1; //文件总字节数 $new_length = $size2 - $range; //获取下次下载的长度 header("HTTP/1.1 206 Partial Content"); header("Content-Length: $new_length"); //输入总长 header("Content-Range: bytes $range-$size2/$size"); //Content-Range: bytes 4908618-4988927/4988928 95%的时候 } else { //第一次连接 $size2 = $size -1; header("Content-Range: bytes 0-$size2/$size"); //Content-Range: bytes 0-4988927/4988928 header("Content-Length: " . $size); //输出总长 } //打开文件 $fp = fopen("$sourceFile", "rb"); //设置指针位置 fseek($fp, $range); //虚幻输出 while (!feof($fp)) { //设置文件最长执行时间 set_time_limit(0); print (fread($fp, 1024 * 8)); //输出文件 flush(); //输出缓冲 ob_flush(); } fclose($fp); exit ();
经过修改后的文件下载,进行函数封装,此下载方法已经测试过,支持大文件的下载及断点续传等功能,并修改一些细节及去除没用的代码:
function downloadBagFile($filePath) { //设置文件最长执行时间和内存 set_time_limit ( 0 ); ini_set ( 'memory_limit', '1024M' ); //检测文件是否存在 if (! is_file ( $filePath )) { die ( "<b>404 File not found!</b>" ); } $filename = basename ( $filePath ); //获取文件名字 //开始写输出头信息 header ( "Cache-Control: public" ); //设置输出浏览器格式 header ( "Content-Type: application/octet-stream" ); header ( "Content-Disposition: attachment; filename=" . $filename ); header ( "Content-Transfer-Encoding: binary" ); header ( "Accept-Ranges: bytes" ); $size = filesize ( $filePath ); $range=0; //如果有$_SERVER['HTTP_RANGE']参数 if (isset ( $_SERVER ['HTTP_RANGE'] )) { /*Range头域 Range头域可以请求实体的一个或者多个子范围。 例如, 表示头500个字节:bytes=0-499 表示第二个500字节:bytes=500-999 表示最后500个字节:bytes=-500 表示500字节以后的范围:bytes=500- 第一个和最后一个字节:bytes=0-0,-1 同时指定几个范围:bytes=500-600,601-999 但是服务器可以忽略此请求头,如果无条件GET包含Range请求头,响应会以状态码206(PartialContent)返回而不是以200 (OK). */ // 断点后再次连接 $_SERVER['HTTP_RANGE'] 的值 bytes=4390912- list ( $a, $range ) = explode ( "=", $_SERVER ['HTTP_RANGE'] ); //if yes, download missing part $size2 = $size - 1; //文件总字节数 $new_length = $size2 - $range; //获取下次下载的长度 header ( "HTTP/1.1 206 Partial Content" ); header ( "Content-Length: {$new_length}" ); //输入总长 header ( "Content-Range: bytes {$range}-{$size2}/{$size}" ); //Content-Range: bytes 4908618-4988927/4988928 95%的时候 } else { //第一次连接 $size2 = $size - 1; header ( "Content-Range: bytes 0-{$size2}/{$size}" ); //Content-Range: bytes 0-4988927/4988928 header ( "Content-Length: " . $size ); //输出总长 } //打开文件 $fp = fopen ( "{$filePath}", "rb" ); //设置指针位置 fseek ( $fp, $range ); //虚幻输出 while ( ! feof ( $fp ) ) { print ( fread ( $fp, 1024 * 8 ) ); //输出文件 flush (); //输出缓冲 ob_flush (); } fclose ( $fp ); exit (); } $filePath = "E:/soft/PyCharm_setup.zip"; downloadBagFile ( $filePath );
已经测试过了,果然好用,准备用在项目中。
转载请注明:谷谷点程序 » php超大文件及断点续传下载函数