在PHP中使用X-SendFile头让文件下载更快


Posted in PHP onJune 01, 2014

一般来说, 我们可以通过直接让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 . '"');
    } else {
     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直接把文件发送给用户呢?

今天, 我看到了一个有意思的文章: 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 = 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也有类似的模块, 大家有兴趣的可以去找找看

PHP 相关文章推荐
php中iconv函数使用方法
May 24 PHP
php生成随机数或者字符串的代码
Sep 05 PHP
Look And Say 序列php实现代码
May 22 PHP
PHP数据库操作之基于Mysqli的数据库操作类库
Apr 19 PHP
PHP获取文件的MD5值并判断是否被修改的例子
Jun 19 PHP
laravel安装和配置教程
Oct 29 PHP
9条PHP编程小知识及易犯的小错误
Jan 22 PHP
php集成套件服务器xampp安装使用教程(适合第一次玩PHP的新手)
Jun 03 PHP
php正则匹配文章中的远程图片地址并下载图片至本地
Sep 29 PHP
PHP利用超级全局变量$_POST来接收表单数据的实例
Nov 05 PHP
php伪静态验证码不显示的解决方案
Sep 26 PHP
laravel框架邮箱认证实现方法详解
Nov 22 PHP
PHP is_subclass_of函数的一个BUG和解决方法
Jun 01 #PHP
PHP中数组的分组排序实例
Jun 01 #PHP
php_screw安装使用教程(另一个PHP代码加密实现)
May 29 #PHP
PHP Curl出现403错误的解决办法
May 29 #PHP
PHP的foreach中使用引用时需要注意的一个问题和解决方法
May 29 #PHP
神盾加密解密教程(一)PHP变量可用字符
May 28 #PHP
CI框架开发新浪微博登录接口源码完整版
May 28 #PHP
You might like
用PHP写的MySQL数据库用户认证系统代码
2007/03/22 PHP
Excel数据导入Mysql数据库的实现代码
2008/06/05 PHP
教你如何开启shopnc b2b2c 伪静态
2014/10/21 PHP
php将textarea数据提交到mysql出现很多空格的解决方法
2014/12/19 PHP
简单解决微信文章图片防盗链问题
2016/12/17 PHP
PHP设计模式之装饰器模式定义与用法简单示例
2018/08/13 PHP
仿百度联盟对联广告实现代码
2014/08/30 Javascript
node.js中的fs.realpath方法使用说明
2014/12/16 Javascript
JavaScript焦点事件、鼠标事件和滚轮事件使用详解
2016/01/15 Javascript
JavaScript小技巧整理篇(非常全)
2016/01/26 Javascript
JavaScript中日期函数的相关操作知识
2016/08/03 Javascript
Angular2 组件间通过@Input @Output通讯示例
2017/08/24 Javascript
详解nodejs中express搭建权限管理系统
2017/09/15 NodeJs
详解如何在angular2中获取节点
2017/11/23 Javascript
浅析Visual Studio Code断点调试Vue
2018/02/27 Javascript
Vue传参一箩筐(页面、组件)
2019/04/04 Javascript
开发中常用的25个JavaScript单行代码(小结)
2019/06/28 Javascript
layui实现下拉框三级联动
2019/07/26 Javascript
vue props 单项数据流实例分享
2020/02/16 Javascript
查看Python安装路径以及安装包路径小技巧
2015/04/28 Python
python3+PyQt5实现文档打印功能
2018/04/24 Python
Python Django框架模板渲染功能示例
2019/11/08 Python
Python 之 Json序列化嵌套类方式
2020/02/27 Python
Django实现列表页商品数据返回教程
2020/04/03 Python
Python .py生成.pyd文件并打包.exe 的注意事项说明
2021/03/04 Python
H5最强接口之canvas实现动态图形功能
2019/05/31 HTML / CSS
美国学校校服,儿童和婴儿服装:Cookie’s Kids
2016/10/14 全球购物
美国健康和保健平台:healtop
2020/07/02 全球购物
本科生详细的自我评价
2013/09/19 职场文书
学校食堂食品安全责任书
2014/07/28 职场文书
开票员岗位职责
2015/02/12 职场文书
铁人观后感
2015/06/16 职场文书
2016小学优秀教师先进事迹材料
2016/02/26 职场文书
市场营销计划书
2019/04/24 职场文书
Go语言中break label与goto label的区别
2021/04/28 Golang
SQL Server携程核心系统无感迁移到MySQL实战
2022/06/01 SQL Server