最近导出的时候出现一个php内存溢出的问题,原因就是在于下载的时候读取生成的临时文件过大,PHP内存无法容纳,一开如是想到更改PHP内存限制,但是这个只是一个缓兵之计,于是想到了另外一个方法是把文件分次读取,并下载。
以下是源代码:
<?php <BR>$sourceFile = "1.tmp"; //要下载的临时文件名 <BR>$outFile = "用户订单.xls"; //下载保存到客户端的文件名 <BR>$file_extension = strtolower(substr(strrchr($sourceFile, "."), 1)); //获取文件扩展名 <BR>//echo $sourceFile; <BR>if (!ereg("[tmp|txt|rar|pdf|doc]", $file_extension))exit ("非法资源下载"); <BR>//检测文件是否存在 <BR>if (!is_file($sourceFile)) { <BR>die("<b>404 File not found!</b>"); <BR>} <BR>$len = filesize($sourceFile); //获取文件大小 <BR>$filename = basename($sourceFile); //获取文件名字 <BR>$outFile_extension = strtolower(substr(strrchr($outFile, "."), 1)); //获取文件扩展名 <BR>//根据扩展名 指出输出浏览器格式 <BR>switch ($outFile_extension) { <BR>case "exe" : <BR>$ctype = "application/octet-stream"; <BR>break; <BR>case "zip" : <BR>$ctype = "application/zip"; <BR>break; <BR>case "mp3" : <BR>$ctype = "audio/mpeg"; <BR>break; <BR>case "mpg" : <BR>$ctype = "video/mpeg"; <BR>break; <BR>case "avi" : <BR>$ctype = "video/x-msvideo"; <BR>break; <BR>default : <BR>$ctype = "application/force-download"; <BR>} <BR>//Begin writing headers <BR>header("Cache-Control:"); <BR>header("Cache-Control<b>/本文来源gao@!dai!ma.com搞$$代^@码5网@</b><strong>搞代gaodaima码</strong>: public"); <BR>//设置输出浏览器格式 <BR>header("Content-Type: $ctype"); <BR>header("Content-Disposition: attachment; filename=" . $outFile); <BR>header("Accept-Ranges: bytes"); <BR>$size = filesize($sourceFile); <BR>//如果有$_SERVER['HTTP_RANGE']参数 <BR>if (isset ($_SERVER['HTTP_RANGE'])) { <BR>/*Range头域 Range头域可以请求实体的一个或者多个子范围。 <BR>例如, <BR>表示头500个字节:bytes=0-499 <BR>表示第二个500字节:bytes=500-999 <BR>表示最后500个字节:bytes=-500 <BR>表示500字节以后的范围:bytes=500- <BR>第一个和最后一个字节:bytes=0-0,-1 <BR>同时指定几个范围:bytes=500-600,601-999 <BR>但是服务器可以忽略此请求头,如果无条件GET包含Range请求头,响应会以状态码206(PartialContent)返回而不是以200 (OK)。 <BR>*/ <BR>// 断点后再次连接 $_SERVER['HTTP_RANGE'] 的值 bytes=4390912- <BR>list ($a, $range) = explode("=", $_SERVER['HTTP_RANGE']); <BR>//if yes, download missing part <BR>str_replace($range, "-", $range); //这句干什么的呢。。。。 <BR>$size2 = $size -1; //文件总字节数 <BR>$new_length = $size2 - $range; //获取下次下载的长度 <BR>header("HTTP/1.1 206 Partial Content"); <BR>header("Content-Length: $new_length"); //输入总长 <BR>header("Content-Range: bytes $range$size2/$size"); //Content-Range: bytes 4908618-4988927/4988928 95%的时候 <BR>} else { <BR>//第一次连接 <BR>$size2 = $size -1; <BR>header("Content-Range: bytes 0-$size2/$size"); //Content-Range: bytes 0-4988927/4988928 <BR>header("Content-Length: " . $size); //输出总长 <BR>} <BR>//打开文件 <BR>$fp = fopen("$sourceFile", "rb"); <BR>//设置指针位置 <BR>fseek($fp, $range); <BR>//虚幻输出 <BR>while (!feof($fp)) { <BR>//设置文件最长执行时间 <BR>set_time_limit(0); <BR>print (fread($fp, 1024 * 8)); //输出文件 <BR>flush(); //输出缓冲 <BR>ob_flush(); <BR>} <BR>fclose($fp); <BR>exit (); <BR>