用戶端從服務端下載檔案的流程分析:
瀏覽器發送一個請求,請求訪問伺服器中的某個網頁(如:down.php),該網頁的代碼如下。
伺服器接受到該請求以後,馬上運行該down.php檔案
運行該檔案的時候,必然要把將要被下載的檔案讀入記憶體當中(這裡是聖誕狂歡.jpg這張圖片),這裡通過fopen()函數完成該動作
代碼:
| 代碼如下 |
複製代碼 |
<?php header("Content-type:text/html;charset=utf-8"); // $file_name="cookie.jpg"; $file_name="聖誕狂歡.jpg"; //用以解決中文不能顯示出來的問題 $file_name=iconv("utf-8","gb2312",$file_name); $file_sub_path=$_SERVER['DOCUMENT_ROOT']."marcofly/phpstudy/down/down/"; $file_path=$file_sub_path.$file_name; //首先要判斷給定的檔案存在與否 if(!file_exists($file_path)){ echo "沒有該檔案檔案"; return ; } $fp=fopen($file_path,"r"); $file_size=filesize($file_path); //下載檔案需要用到的頭 Header("Content-type: application/octet-stream"); Header("Accept-Ranges: bytes"); Header("Accept-Length:".$file_size); Header("Content-Disposition: attachment; filename=".$file_name); $buffer=1024; $file_count=0; //向瀏覽器返回資料 while(!feof($fp) && $file_count<$file_size){ $file_con=fread($fp,$buffer); $file_count+=$buffer; echo $file_con; } fclose($fp); ?> |
header("Content-type:text/html;charset=utf-8")的作用:在伺服器響應瀏覽器的請求時,告訴瀏覽器以編碼格式為UTF-8的編碼顯示該內容
關於file_exists()函數不支援中文路徑的問題:因為php函數比較早,不支援中文,所以如果被下載的檔案名稱是中文的話,需要對其進行字元編碼轉換,否則file_exists()函數不能識別,可以使用iconv()函數進行編碼轉換
$file_sub_path() 我使用的是絕對路徑,執行效率要比相對路徑高
Header("Content-type: application/octet-stream")的作用:通過這句代碼用戶端瀏覽器就能知道服務端返回的檔案形式
Header("Accept-Ranges: bytes")的作用:告訴用戶端瀏覽器返回的檔案大小是按照位元組進行計算的
Header("Accept-Length:".$file_size)的作用:告訴瀏覽器返回的檔案大小
Header("Content-Disposition: attachment; filename=".$file_name)的作用:告訴瀏覽器返回的檔案的名稱
以上四個Header()是必需的
fclose($fp)可以把緩衝區內最後剩餘的資料輸出到磁碟檔案中,並釋放檔案指標和有關的緩衝區
注意:任何有關從伺服器下載的檔案操作,必然需要先在服務端將檔案讀入記憶體當中
現在檔案已經在記憶體當中了,這是需要從記憶體當中讀取檔案,通過fread()函數完成該動作
需要注意的是,如果檔案較大,檔案應該是被分成多段返回給用戶端的,並不是等檔案在服務端全部讀取完畢後,一次性返回給用戶端,因為這樣子會增加伺服器的負荷。