客戶端從服務端下載文件的流程分析:
- 瀏覽器發送一個請求,請求訪問服務器中的某個網頁(如:down.php),該網頁的代碼如下。
- 服務器接受到該請求以后,馬上運行該down.php文件
- 運行該文件的時候,必然要把將要被下載的文件讀入內存當中(這里是聖誕狂歡.jpg這張圖片),這里通過fopen()函數完成該動作
注意:任何有關從服務器下載的文件操作,必然需要先在服務端將文件讀入內存當中
- 現在文件已經在內存當中了,這是需要從內存當中讀取文件,通過fread()函數完成該動作
- 需要注意的是,如果文件較大,文件應該是被分成多段返回給客戶端的,並不是等文件在服務端全部讀取完畢后,一次性返回給客戶端,因為這樣子會增加服務器的負荷。
所以我們需要在php代碼中設置一次讀取的字節數,比如我在下面的代碼中通過$buffer=1024設置一次讀取的字節數,每讀取一次,就輸出數據(即返回給瀏覽器)
流程圖:
1 代碼: 2 <?php 3 header("Content-type:text/html;charset=utf-8"); 4 // $file_name="cookie.jpg"; 5 $file_name="聖誕狂歡.jpg"; 6 //用以解決中文不能顯示出來的問題 7 $file_name=iconv("utf-8","gb2312",$file_name); 8 $file_sub_path=$_SERVER['DOCUMENT_ROOT']."marcofly/phpstudy/down/down/"; 9 $file_path=$file_sub_path.$file_name; 10 //首先要判斷給定的文件存在與否 11 if(!file_exists($file_path)){ 12 echo "沒有該文件文件"; 13 return ; 14 } 15 $fp=fopen($file_path,"r"); 16 $file_size=filesize($file_path); 17 //下載文件需要用到的頭 18 Header("Content-type: application/octet-stream"); 19 Header("Accept-Ranges: bytes"); 20 Header("Accept-Length:".$file_size); 21 Header("Content-Disposition: attachment; filename=".$file_name); 22 $buffer=1024; 23 $file_count=0; 24 //向瀏覽器返回數據 25 while(!feof($fp) && $file_count<$file_size){ 26 $file_con=fread($fp,$buffer); 27 $file_count+=$buffer; 28 echo $file_con; 29 } 30 fclose($fp); 31 ?>
幾點注意事項:
- 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)可以把緩沖區內最后剩余的數據輸出到磁盤文件中,並釋放文件指針和有關的緩沖區