讓 PHP 更快的提供文件下載

fmms 12年前發布 | 16K 次閱讀 PHP PHP開發

一般來說, 我們可以通過直接讓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 = urlencode($filename);
    $encoded_filename = str_replace("+", "%20", $encoded_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 . '"');
    }

    header('Content-Disposition: attachment; filename="' . $filename . '"');
    header("Content-Length: ". filesize($file));
    readfile($file);</pre> <p></p>

恩, 現在看起來好多了, 不過還有一個問題, 那就是readfile, 雖然PHP的readfile嘗試實現的盡量高效, 不占用PHP本身的內存, 但是實際上它還是需要采用MMAP(如果支持), 或者是一個固定的buffer去循環讀取文件, 直接輸出.

輸出的時候, 如果是Apache + PHP mod, 那么還需要發送到Apache的輸出緩沖區. 最后才發送給用戶. 而對于Nginx + fpm如果他們分開部署的話, 那還會帶來額外的網絡IO.

那么, 能不能不經過PHP這層, 直接讓Webserver直接把文件發送給用戶呢?

今天, 我看到了一個有意思的文章: How I PHP: X-SendFile.

我們可以使用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 = urlencode($filename);
    $encoded_filename = str_replace("+", "%20", $encoded_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 . '"');
    }

    header('Content-Disposition: attachment; filename="' . basename($file) . '"');

    //讓Xsendfile發送文件
    header("X-Sendfile: $file");</pre> <p></p>

X-Sendfile頭將被Apache處理, 并且把響應的文件直接發送給Client.

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

 

 本文由用戶 fmms 自行上傳分享,僅供網友學習交流。所有權歸原作者,若您的權利被侵害,請聯系管理員。
 轉載本站原創文章,請注明出處,并保留原始鏈接、圖片水印。
 本站是一個以用戶分享為主的開源技術平臺,歡迎各類分享!