php文件下載

客戶端從服務端下載文件的流程分析: 

瀏覽器發送一個請求,請求訪問服務器中的某個網頁(如:down.php),該網頁的代碼以下。 
服務器接受到該請求之後,立刻運行該down.php文件 
運行該文件的時候,必然要把將要被下載的文件讀入內存當中(這裏是聖誕狂歡.jpg這張圖片),這裏經過fopen()函數完成該動做 
注意:任何有關從服務器下載的文件操做,必然須要先在服務端將文件讀入內存當中 

如今文件已經在內存當中了,這是須要從內存當中讀取文件,經過fread()函數完成該動做 
須要注意的是,若是文件較大,文件應該是被分紅多段返回給客戶端的,並非等文件在服務端所有讀取完畢後,一次性返回給客戶端,由於這樣子會增長服務器的負荷。 
因此咱們須要在php代碼中設置一次讀取的字節數,好比我在下面的代碼中經過$buffer=1024設置一次讀取的字節數,每讀取一次,就輸出數據(即返回給瀏覽器) 

流程圖:  php

代碼以下:

<?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)能夠把緩衝區內最後剩餘的數據輸出到磁盤文件中,並釋放文件指針和有關的緩衝區
相關文章
相關標籤/搜索