如何解決大文件切片上傳無法處理48MB以上文件的問題?

如何解決大文件切片上傳無法處理48MB以上文件的問題?

突破48MB限制:解決大文件切片上傳難題

大文件切片上傳過程中,遇到48MB以上文件無法處理的情況?本文將分析問題根源,并提供前端、后端及服務器配置方面的解決方案。

問題剖析

上傳大文件時,在第48片(約48MB)之后,后續請求返回500錯誤,即使調整切片大小,也無法突破40MB的限制。這表明問題可能出在服務器端或后端處理邏輯上,需要深入排查。

前端代碼優化

首先,檢查前端代碼。之前的代碼可能在每次發送切片時使用同一個FormData對象,導致數據累積超過服務器限制。

解決方案:每次調用上傳函數時,重新創建FormData對象。這樣,每個請求只包含當前切片數據,避免數據累積。

示例代碼片段 (改進后的sendfile函數):

function sendfile(blob, file) {     if (is_stop == 0) {         var xhr = new XMLHttpRequest();         var form_data = new FormData(); // 重新創建FormData對象          var total_blob_num = Math.ceil(file.size / length);         form_data.append('file', blob);         form_data.append('blob_num', blob_num);         form_data.append('total_blob_num', total_blob_num);         form_data.append('file_name', file.name);          xhr.open('POST', '/upload.php', false);         xhr.onreadystatechange = function () {             // ... (其余代碼保持不變) ...         };         xhr.send(form_data); // 發送請求     } }

后端代碼改進

后端代碼可能因直接在內存中處理大文件而導致內存溢出。建議采用流式處理或臨時文件合并的方式。將每個切片保存到臨時文件,最后再合并成最終文件。

示例代碼片段 (PHP,改進后的fileMerge函數):

private function fileMerge() {     if ($this->blobNum == $this->totalBlobNum) {         $out = fopen($this->filepath . '/' . $this->fileName, 'wb'); // 使用'wb'模式,確保二進制安全         for ($i = 1; $i <= $this->totalBlobNum; $i++) {             $chunk = fopen($this->filepath . '/' . $this->fileName . '__' . $i, 'rb'); // 使用'rb'模式讀取二進制數據             stream_copy_to_stream($chunk, $out); // 使用stream_copy_to_stream高效復制             fclose($chunk);             @unlink($this->filepath . '/' . $this->fileName . '__' . $i);         }         fclose($out);     } }

服務器配置檢查

即使調整了前端和后端代碼,問題可能仍然存在。需要檢查以下服務器配置:

  1. apache的LimitRequestBody設置: 確保此值足夠大,允許上傳大文件。
  2. PHP的upload_max_filesize和post_max_size設置: 這兩個值必須大于上傳文件的大小。
  3. PHP的memory_limit設置: 如果后端在內存中處理文件,需要增加此值。

總結與建議

解決大文件切片上傳問題,需要綜合考慮前端、后端和服務器配置。通過重新創建FormData對象、采用流式處理或臨時文件合并以及調整服務器配置,可以有效解決48MB以上文件上傳的難題,實現高效的大文件上傳功能。

? 版權聲明
THE END
喜歡就支持一下吧
點贊10 分享