讓PHP更快的提供檔案下載
阿新 • • 發佈:2019-01-10
一般來說, 我們可以通過直接讓URL指向一個位於Document Root下面的檔案, 來引導使用者下載檔案.
但是, 這樣做, 就沒辦法做一些統計, 許可權檢查, 等等的工作. 於是, 很多時候, 我們採用讓PHP來做轉發, 為使用者提供檔案下載.
- <?php
- $file = "/tmp/dummy.tar.gz";
- header("Content-type: application/octet-stream");
- header('Content-Disposition: attachment; filename="' . basename($file) . '"');
- header("Content-Length: ". filesize($file));
- readfile($file);
但是這個有一個問題, 就是如果檔案是中文名的話, 有的使用者可能下載後的檔名是亂碼.
於是, 我們做一下修改(參考: :
- <?php
- $file = "/tmp/中文名.tar.gz";
- $filename = basename($file);
- header("Content-type: application/octet-stream");
- //處理中文檔名
- $ua = $_SERVER["HTTP_USER_AGENT"];
- $encoded_filename = rawurlencode($filename);
- if (preg_match("/MSIE/", $ua)) {
- header('Content-Disposition: attachment; filename="' . $encoded_filename . '"');
- } else if (preg_match("/Firefox/", $ua)) {
- header("Content-Disposition: attachment; filename*=\"utf8''" . $filename . '"');
- }
- header('Content-Disposition: attachment; filename="' . $filename . '"');
- }
- header("Content-Length: ". filesize($file));
- readfile($file);
恩, 現在看起來好多了, 不過還有一個問題, 那就是readfile, 雖然PHP的readfile嘗試實現的儘量高效, 不佔用PHP本身的記憶體, 但是實際上它還是需要採用MMAP(如果支援), 或者是一個固定的buffer去迴圈讀取檔案, 直接輸出.
輸出的時候, 如果是Apache + PHP mod, 那麼還需要傳送到Apache的輸出緩衝區. 最後才傳送給使用者. 而對於Nginx + fpm如果他們分開部署的話, 那還會帶來額外的網路IO.
那麼, 能不能不經過PHP這層, 直接讓Webserver直接把檔案傳送給使用者呢?
我們可以使用Apache的module mod_xsendfile, 讓Apache直接傳送這個檔案給使用者:
- <?php
- $file = "/tmp/中文名.tar.gz";
- $filename = basename($file);
- header("Content-type: application/octet-stream");
- //處理中文檔名
- $ua = $_SERVER["HTTP_USER_AGENT"];
- $encoded_filename = rawurlencode($filename);
- if (preg_match("/MSIE/", $ua)) {
- header('Content-Disposition: attachment; filename="' . $encoded_filename . '"');
- } else if (preg_match("/Firefox/", $ua)) {
- header("Content-Disposition: attachment; filename*=\"utf8''" . $filename . '"');
- } else {
- header('Content-Disposition: attachment; filename="' . $filename . '"');
- }
- //讓Xsendfile傳送檔案
- header("X-Sendfile: $file");
X-Sendfile頭將被Apache處理, 並且把響應的檔案直接傳送給Client.
Lighttpd和Nginx也有類似的模組, 大家有興趣的可以去找找看