隨著行動網路時代的到來,大檔案的傳輸需求越來越普遍。而其中,PHP作為一門流行的程式語言,在大檔案上傳和下載方面有著良好的表現。在本文中,我們將學習PHP中的大檔案上傳和下載技術,包括如何處理大檔案、分塊上傳、斷點續傳和非同步下載等重要技術。
一、PHP中處理大文件的方法
在PHP中處理大文件,最重要的一步是規劃合適的記憶體使用方案。我們不能直接將整個大檔案讀入內存,因為這樣會導致內存溢出並可能使整個程式崩潰。那麼,PHP中處理大檔案的方法是什麼呢?
首先,我們可以使用「檔案指標」的概念來解決這個問題。 「檔案指針」是一個在檔案中定位的指針,我們可以透過它來讀取檔案中的某一部分,而不是一次讀入整個檔案。在PHP中,我們可以使用fopen()和fread()函數來開啟和讀取檔案。
其次,我們可以使用「串流」的方法。將大檔案先分割成多個部分,然後一塊一塊地上傳或下載。這樣就可以避免一次讀取整個大文件,也可以確保資料的完整性。
二、大檔案上傳
對於HTTP通信,PHP預設的上傳檔案大小為2MB,因此我們需要更改php.ini設定檔中的upload_max_filesize選項來允許上傳更大的文件。
但即使更改了php.ini文件,我們仍然面臨著上傳大文件時的問題。我們如何在上傳過程中避免記憶體外洩和程式崩潰?答案是使用分塊上傳技術。
所謂分塊上傳,就是將大檔案分割成固定大小的區塊,然後一塊一塊地上傳,直到上傳完成。這種方法可以避免一次性讀取整個文件,減輕伺服器記憶體壓力。
具體實作如下:
前端程式碼:
<input type="file" name="file" id="file">
var chunkSize = 1 * 1024 * 1024; //每一块的大小为1MB var file = document.getElementById('file').files[0]; var name = file.name; var size = file.size; var chunkNum = Math.ceil(size / chunkSize); //总块数 for(var i=0;i<chunkNum;i++){ var start = i*chunkSize; var end = Math.min(size,start+chunkSize); var chunk = file.slice(start,end); //将文件切成每个chunkSize大小的块 //使用FormData上传 var formData = new FormData(); formData.append('chunk', chunk); formData.append('filename', i+'_'+name); //在文件名前加上块的编号 //发起已经分割好的块的上传请求 send(chunkUrl,formData); }
後端程式碼:
$chunk = $_FILES['chunk']; //获取文件块 $filename = $_POST['filename']; //获取文件名 //上传文件块 move_uploaded_file( $chunk['tmp_name'] , $upload.$filename); echo "success";
以上程式碼實現了將大檔案分成若干個1MB大小的塊來上傳,從而分散壓力。
另一個值得注意的問題是,如果上傳一個大檔案時意外取消了,我們如何在其中斷的位置繼續上傳?
答案是使用斷點續傳技術。我們可以在上傳檔案的同時記錄檔案上傳的進度,然後在使用者再次上傳時,將上次上傳未完成的檔案區塊快取起來,以便後續再次上傳。
具體實作如下:
前端程式碼:
var chunkSize = 1 * 1024 * 1024; //每一块的大小为1MB var file = document.getElementById('file').files[0]; var name = file.name; var size = file.size; var chunkNum = Math.ceil(size / chunkSize); //总块数 for(var i=0;i<chunkNum;i++){ var start = i*chunkSize; var end = Math.min(size,start+chunkSize); var chunk = file.slice(start,end); //将文件切成每个chunkSize大小的块 //使用FormData上传 var formData = new FormData(); formData.append('chunk', chunk); formData.append('filename', i+'_'+name); //在文件名前加上块的编号 formData.append('chunkNum',chunkNum); //总块数 formData.append('index',i); //当前块的编号 //将文件分块上传 send(chunkUrl,formData,function(){ uploaded[i] = 1; //标记该块已上传 var uploadedAll = uploaded.every(v => v == 1); //检查是否所有块都已上传 //如果所有块都已上传,则在服务端组装文件 if(uploadedAll){ var formData = new FormData(); formData.append('name', name); formData.append('chunkNum',chunkNum); //总块数 //发起文件组装请求 send(mergeUrl,formData,function(response){ console.log(response); }); } }); }
後端程式碼:
$chunk = $_FILES['chunk']; //获取文件块 $filename = $_POST['filename']; //获取文件名 $chunkNum = $_POST['chunkNum']; //获取文件总块数 $index = $_POST['index']; //获取当前块的编号 //上传文件块 move_uploaded_file( $chunk['tmp_name'] , $upload.$filename); //上传完成后检查所有块是否已上传 $uploaded = $this->isUploaded($upload,$chunkNum); if($uploaded){ $this->mergeFile($upload,$chunkNum,$name); //组装文件 } echo "success";
以上程式碼實作了將檔案分片上傳,並在客戶端標記上傳的進度,以及在服務端檢查上傳是否完成。當上傳完成時,將上傳的所有檔案區塊進行拼接。
三、大檔案下載
在PHP中進行大檔案下載時,我們同樣需要解決記憶體外洩和程式崩潰的問題。為此,我們可以使用非同步下載技術。
非同步下載是指將下載請求放在單獨的執行緒中進行,以避免阻塞伺服器主執行緒。在PHP中,我們可以使用exec()函數來啟動單獨的執行緒。
具體實作如下:
前端程式碼:
<a href="#" onclick="download()">下载</a>
function download(){ var filename = 'test.zip'; //准备下载文件名 //发起异步下载请求 var ajax = new XMLHttpRequest(); ajax.open('GET', '/download.php?filename='+filename, true); ajax.send(); }
後端程式碼:
$action = isset($_GET['action']) ? $_GET['action'] : 'download'; //获取下载动作 $filename = $_GET['filename']; //获取文件名 if($action == 'download'){ //启动异步下载 exec('php download.php '.$filename.' > /dev/null &'); }else{ //文件下载 header("Content-type: application/octet-stream"); header("Content-Disposition: attachment; filename=".$filename); header("Accept-Ranges: bytes"); $file = fopen($download.$filename,"r"); if($file){ while(!feof($file)){ print(fread($file, 1024*8 )); flush(); ob_flush(); } } fclose($file); }
以上程式碼實現了在客戶端發起非同步下載請求,後端使用exec()函數啟動單獨的執行緒進行下載動作,從而避免了主執行緒被阻塞的問題。同時,在檔案下載時也解決了記憶體問題,使用分塊下載並輸出到緩衝區。
四、總結
在本文中,我們學習了PHP中的大檔案上傳和下載技術,主要包括如何處理大檔案、分塊上傳、斷點續傳和非同步下載等技術。透過本文的指南,我們可以更掌握PHP中處理大檔案的技術,並在實際的專案中運用這些技術來實現大檔案的高效傳輸。
以上是PHP中的大檔案上傳和下載技術指南的詳細內容。更多資訊請關注PHP中文網其他相關文章!