在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中利用XML技术构造远程服务(上)
Oct 09 PHP
php中判断一个字符串包含另一个字符串的方法
Mar 19 PHP
两个强悍的php 图像处理类1
Jun 15 PHP
PHP获取163、gmail、126等邮箱联系人地址【已测试2009.10.10】
Oct 11 PHP
php Static关键字实用方法
Jun 04 PHP
php实现递归抓取网页类实例
Apr 03 PHP
php基本函数汇总
Jul 09 PHP
PHP简单获取及判断提交来源的方法
Apr 22 PHP
动态表单验证的操作方法和TP框架里面的ajax表单验证
Jul 19 PHP
PHP使用栈解决约瑟夫环问题算法示例
Aug 27 PHP
PHPstorm激活码2020年5月13日亲测有效
Sep 17 PHP
php命令行模式代码实例详解
Feb 26 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模板类代码
2008/09/07 PHP
PHP 判断变量类型实现代码
2009/10/23 PHP
PHP中的替代语法介绍
2015/01/09 PHP
laravel 解决多库下的DB::transaction()事务失效问题
2019/10/21 PHP
利用jQuery的$.event.fix函数统一浏览器event事件处理
2009/12/21 Javascript
js 模拟气泡屏保效果代码
2010/07/10 Javascript
Wordpress ThickBox 添加“查看原图”效果代码
2010/12/11 Javascript
用js判断页面刷新或关闭的方法(onbeforeunload与onunload事件)
2012/06/22 Javascript
利用JS判断用户是否上网(连接网络)
2013/12/23 Javascript
jquery选择器原理介绍($()使用方法)
2014/03/25 Javascript
Node.js中使用mongoskin操作mongoDB实例
2014/09/28 Javascript
如何解决easyui自定义标签 datagrid edit combobox 手动输入保存不上
2015/12/26 Javascript
[原创]Bootstrap 中下拉菜单修改成鼠标悬停直接显示
2016/04/14 Javascript
JavaScript手机振动API
2016/06/11 Javascript
JS导出PDF插件的方法(支持中文、图片使用路径)
2016/07/12 Javascript
JS 调用微信扫一扫功能
2016/12/22 Javascript
jQuery Masonry瀑布流布局神器使用详解
2017/05/25 jQuery
vue项目打包部署_nginx代理访问方法详解
2018/09/20 Javascript
微信小程序 简易计算器实现代码实例
2019/09/02 Javascript
聊聊vue 中的v-on参数问题
2021/01/29 Vue.js
使用python调用浏览器并打开一个网址的例子
2014/06/05 Python
python访问类中docstring注释的实现方法
2015/05/04 Python
举例详解Python中threading模块的几个常用方法
2015/06/18 Python
深入理解Python中变量赋值的问题
2017/01/12 Python
Python新手们容易犯的几个错误总结
2017/04/01 Python
python3库numpy数组属性的查看方法
2018/04/17 Python
详解Python最长公共子串和最长公共子序列的实现
2018/07/07 Python
python实现自动解数独小程序
2019/01/21 Python
Python二次规划和线性规划使用实例
2019/12/09 Python
Python 抓取数据存储到Redis中的操作
2020/07/16 Python
eBay澳大利亚站:eBay.com.au
2018/02/02 全球购物
幼儿园秋游感想
2014/03/12 职场文书
手机银行营销方案
2014/03/14 职场文书
保险公司早会主持词
2014/03/22 职场文书
职务说明书范文
2014/05/07 职场文书
工作证明格式范文
2015/06/15 职场文书