簡單的檔案下載只需要使用HTML的串連標記<a>,並將屬性href的URL值指定為下載的檔案即可。代碼如下所示:
| 代碼如下 |
複製代碼 |
<a href=”http://www.111cn.net /download/book.rar”>下載檔案</a> |
如果通過上面的代碼實現檔案下載,只能處理一些瀏覽器不能預設識別的MIME類型檔案,例如當訪問book.rar檔案時,瀏覽器並沒有直接開啟,而是彈出一個下載提示框,提示使用者“下載”還是“開啟”等處理方式。但如果需要下載尾碼名為.html的網頁檔案、圖片檔案及PHP程式指令檔等,使用這種串連形式,則會將檔案內容直接輸出到瀏覽器中,並不會提示使用者下載。
為了提高檔案的安全性,不希望在<a>標籤中給出檔案的連結,則必須向瀏覽器發送必要的頭資訊,以通知瀏覽器將要進行下載檔案的處理。PHP使用header()函數發送網頁的頭部資訊給瀏覽器,該函數接收一個頭資訊的字串作為參數。檔案下載需要發送的頭資訊包括以下三部分,通過調用三次header()函數完成。以下載圖片test.gif為例,需要發送的頭資訊的代碼如下所示:
| 代碼如下 |
複製代碼 |
header(‘Content-Type:imge/gif’); //發送指定檔案MIME類型的頭資訊 header(‘Content-Disposition:attachment; filename=”test.gif”‘); //發送描述檔案的頭資訊,附件和檔案名稱 header(‘Content-Length:3390′); //發送指定檔案大小的資訊,單位位元組 |
如果使用header()函數向瀏覽器發送了這三行頭資訊,圖片test.gif就不會直接在瀏覽器中顯示,而讓瀏覽器將該檔案形成下載的形式。在函數header()中,“Content-Type”指定了檔案的MIME類型,“Content_Disposition”用於檔案的描述,值“attachment; filename=”test.gif””說明這是一個附件,並且指定了下載後的檔案名稱,“Content_Length”則給出了被下載檔案的大小。
設定完頭部資訊以後,需要將檔案的內容輸出到瀏覽器,以便進行下載。可以使用PHP中的檔案系統函數將檔案內容讀取出來後,直接輸出給瀏覽器。最方便的是使用readfile()函數,將檔案內容讀取出來直接輸出。下載檔案test.gif的代碼如下所示:
| 代碼如下 |
複製代碼 |
<?php $filename = "test.gif"; header('Content-Type:image/gif'); //指定下載檔案類型 header('Content-Disposition: attachment; filename="'.$filename.'"'); //指定下載檔案的描述 header('Content-Length:'.filesize($filename)); //指定下載檔案的大小 //將檔案內容讀取出來並直接輸出,以便下載 readfile($filename); ?> |
上面如果碰到中文名字就會無法正常下載了,對於中文名字下載檔案我又找到一個檔案下載執行個體代碼
| 代碼如下 |
複製代碼 |
<?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)可以把緩衝區內最後剩餘的資料輸出到磁碟檔案中,並釋放檔案指標和有關的緩衝區