1. 程式人生 > >讓PHP更快的提供檔案下載

讓PHP更快的提供檔案下載


一般來說, 我們可以通過直接讓URL指向一個位於Document Root下面的檔案, 來引導使用者下載檔案.

但是, 這樣做, 就沒辦法做一些統計, 許可權檢查, 等等的工作. 於是, 很多時候, 我們採用讓PHP來做轉發, 為使用者提供檔案下載.

  1. <?php
  2.     $file = "/tmp/dummy.tar.gz";
  3.     header("Content-type: application/octet-stream");
  4.     header('Content-Disposition: attachment; filename="' . basename($file) . '"');
  5.     header("Content-Length: ". filesize($file));
  6.     readfile($file);

但是這個有一個問題, 就是如果檔案是中文名的話, 有的使用者可能下載後的檔名是亂碼.

於是, 我們做一下修改(參考: :

  1. <?php
  2.     $file = "/tmp/中文名.tar.gz";
  3.     $filename = basename($file);
  4.     header("Content-type: application/octet-stream");
  5.     //處理中文檔名
  6.     $ua = $_SERVER["HTTP_USER_AGENT"];
  7.     $encoded_filename = rawurlencode($filename);
  8.     if (preg_match("/MSIE/", $ua)) {
  9.      header('Content-Disposition: attachment; filename="' . $encoded_filename . '"');
  10.     } else if (preg_match("/Firefox/", $ua)) {
  11.      header("Content-Disposition: attachment; filename*=\"utf8''" . $filename . '"');
  12.     }
    else {
  13.      header('Content-Disposition: attachment; filename="' . $filename . '"');
  14.     }
  15.     header("Content-Length: ". filesize($file));
  16.     readfile($file);

恩, 現在看起來好多了, 不過還有一個問題, 那就是readfile, 雖然PHP的readfile嘗試實現的儘量高效, 不佔用PHP本身的記憶體, 但是實際上它還是需要採用MMAP(如果支援), 或者是一個固定的buffer去迴圈讀取檔案, 直接輸出.

輸出的時候, 如果是Apache + PHP mod, 那麼還需要傳送到Apache的輸出緩衝區. 最後才傳送給使用者. 而對於Nginx + fpm如果他們分開部署的話, 那還會帶來額外的網路IO.

那麼, 能不能不經過PHP這層, 直接讓Webserver直接把檔案傳送給使用者呢?

我們可以使用Apache的module mod_xsendfile, 讓Apache直接傳送這個檔案給使用者:

  1. <?php
  2.     $file = "/tmp/中文名.tar.gz";
  3.     $filename = basename($file);
  4.     header("Content-type: application/octet-stream");
  5.     //處理中文檔名
  6.     $ua = $_SERVER["HTTP_USER_AGENT"];
  7.     $encoded_filename = rawurlencode($filename);
  8.     if (preg_match("/MSIE/", $ua)) {
  9.      header('Content-Disposition: attachment; filename="' . $encoded_filename . '"');
  10.     } else if (preg_match("/Firefox/", $ua)) {
  11.      header("Content-Disposition: attachment; filename*=\"utf8''" . $filename . '"');
  12.     } else {
  13.      header('Content-Disposition: attachment; filename="' . $filename . '"');
  14.     }
  15.     //讓Xsendfile傳送檔案
  16.     header("X-Sendfile: $file");

X-Sendfile頭將被Apache處理, 並且把響應的檔案直接傳送給Client.

Lighttpd和Nginx也有類似的模組, 大家有興趣的可以去找找看 :)