先上一段代碼,支持大文件下載和斷點續傳,代碼來源互聯網。nginx
set_time_limit(0); // 省略取文件路徑的過程,這裏直接是文件完整路徑 $filePath = get_save_path() . $File['save_name']; $filePath = realpath($filePath); $outFileExtension = strtolower(substr(strrchr($filePath, "."), 1)); //獲取文件擴展名 //根據擴展名 指出輸出瀏覽器格式 switch ($outFileExtension) { 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"; } header("Cache-Control:"); header("Cache-Control: public"); //設置輸出瀏覽器格式 header("Content-Type: $ctype"); header("Content-Disposition: attachment; filename=" . basename($filePath)); header("Accept-Ranges: bytes"); $size = filesize($filePath); //若是有$_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("$filePath", "rb"); //設置指針位置 if (!empty($range)) { fseek($fp, $range); } //虛幻輸出 while (!feof($fp)) { print (fread($fp, 1024 * 8)); //輸出文件 flush(); //輸出緩衝 ob_flush(); } fclose($fp); exit ();
代碼有詳細的解釋,也很清楚,可是在實際使用時仍是小文件能夠下載,大文件只能下載前半部分或者出現文件已損壞的狀況。查看nginx日誌發現以下報錯瀏覽器
2018/08/01 07:43:20 [crit] 13906#0: *1479 open() "/usr/local/nginx/fastcgi_temp/0/02/0000000020" failed (13: Permission denied) while reading upstream,
原來在下載大文件時,文件大小超過配置的proxy_temp_file_write_size值時,nginx會將文件寫入到臨時目錄下,若是該目錄沒有權限,就寫不了,那下載只能下載緩衝區的內容了。
覈實/usr/local/nginx/fastcgi_temp/目錄的權限分組,並不在nginx運行帳號組下,即然知道了問題緣由,那就好辦了。給予寫權限,或者將目錄改成nginx運行帳號組下就OK了服務器
cd /usr/local/nginx/ chmod -R 766 fastcgi_temp/ 或者 chown -R nginx:nginx fastcgi_temp/ #nginx根據各自狀況多是不能的帳戶
解決PHP超大文件下載,斷點續傳下載的方法詳解