[TOC]
# PHP实现
~~~
$pathToFile = 'myfile.csv';//文件绝对路径
$downloadName = 'downloadFile.csv';//下载后的文件名
//输入文件标签
Header("Content-type: application/octet-stream");
Header("Accept-Ranges: bytes");
Header("Accept-Length: " . filesize($pathToFile));
Header("Content-Disposition: filename=" . $downloadName);
//输出文件内容
$file = fopen($pathToFile, "r");
echo fread($file, filesize($pathToFile));
fclose($file);
//或
//readfile($pathToFile);
~~~
其中fread()与readfile()的区别可以参考
https://segmentfault.com/q/1010000000647226
但是有时候为了节省带宽,避免瞬时流量过大而造成网络堵塞,就要考虑下载限速的问题
# 下载文件限速
~~~
$pathToFile = 'myfile.csv';//文件绝对路径
$downloadName = 'downloadFile.csv';//下载后的文件名
$download_rate = 30;// 设置下载速率(30 kb/s)
if (file_exists($pathToFile) && is_file($pathToFile)) {
header('Cache-control: private');// 发送 headers
header('Content-Type: application/octet-stream');
header('Content-Length: ' . filesize($pathToFile));
header('Content-Disposition: filename=' . $downloadName);
flush();// 刷新内容
$file = fopen($pathToFile, "r");
while (!feof($file)) {
print fread($file, round($download_rate * 1024));// 发送当前部分文件给浏览者
flush();// flush 内容输出到浏览器端
sleep(1);// 终端1秒后继续
}
fclose($file);// 关闭文件流
} else {
abort(500, '文件' . $pathToFile . '不存在');
}
~~~
此时出现一个问题,当$download_rate>1kb时,文件正常下载;当$download_rate<1kb时,文件要等一会儿才下载,究其原因是因为buffer的问题。
* buffer是一个内存地址空间,Linux系统默认大小一般为4096(1kb),即一个内存页。主要用于存储速度不同步的设备或者优先级不同的设备之间传办理数据的区域。举个例子,你打开文本编辑器编辑一个文件的时候,你每输入一个字符,操作系统并不会立即把这个字符直接写入到磁盘,而是先写入到buffer,当写满了一个buffer的时候,才会把buffer中的数据写入磁盘。同样的道理,当执行echo,print的时候,输出并没有立即通过tcp传给客户端浏览器显示,而是将数据写入php buffer。php output_buffering机制,意味在tcp buffer之前,建立了一新的队列,数据必须经过该队列。当一个php buffer写满的时候,脚本进程会将php buffer中的输出数据交给系统内核交由tcp传给浏览器显示。所以,数据会依次写到这几个地方echo/pring -> php buffer -> tcp buffer -> browser。资料:http://blog.csdn.net/superhosts/article/details/42292053
* 在没有开启缓存时,脚本输出的内容都在服务器端处于等待输出的状态,flush()可以将等待输出的内容立即发送到客户端。
* 开启缓存后,脚本输出的内容存入了输出缓存中,这时没有处于等待输出状态的内容,你直接使用flush()不会向客户端发出任何内容。而ob_flush()的作用就是将本来存在输出缓存中的内容取出来,设置为等待输出状态,但不会直接发送到客户端,这时你就需要先使用ob_flush()再使用flush(),客户端才能立即获得脚本的输出。
* 以及这篇文章同样讲述了ob_flush()和flush()的区别http://www.laruence.com/2010/04/15/1414.html
但是这种方法将文件内容从磁盘经过一个固定的 buffer 去循环读取到内存,再发送给前端 web 服务器,最后才到达用户。当需要下载的文件很大的时候,这种方式将消耗大量内存,甚至引发 php 进程超时或崩溃,接下来就使用到X-Sendfile。
# X-Sendfile
X-Sendfile 是一种将文件下载请求由后端应用转交给前端 web
服务器处理的机制,它可以消除后端程序既要读文件又要处理发送的压力,从而显著提高服务器效率,特别是处理大文件下载的情形下。
我是用的nginx,所以apache请参考https://tn123.org/mod_xsendfile/
①首先在配置文件中添加
~~~
location /download/ {
internal;
root /some/path;//绝对路径
}
~~~
internal 表示这个路径只能在 Nginx 内部访问,不能用浏览器直接访问防止未授权的下载
注意添加在location / {...}的前面
这样你在代码中使用时,文件路径就可以写成“/download/myfile.csv”
②重启Nginx,写代码
~~~
$pathToFile = 'myfile.csv';//文件绝对路径
$downloadName = 'downloadFile.csv';//下载后的文件名
$download_rate = 30;// 设置下载速率(30 kb/s)
if (file_exists($pathToFile) && is_file($pathToFile)) {
return (new Response())->withHeaders([
'Content-Type' => 'application/octet-stream',
'Content-Disposition' => 'attachment;filename=' . $downloadName,
'X-Accel-Redirect' => $pathToFile,//让Xsendfile发送文件
'X-Sendfile' => $pathToFile,
'X-Accel-Limit-Rate' => $download_rate,
]);
}else {
abort(500, '文件' . $pathToFile . '不存在');
}
~~~
![](https://box.kancloud.cn/7d916847a124b6967e6dfb0a66462408_1562x482.jpg)
- OAuth
- 简介
- 步骤
- 单点登录
- .user.ini
- 时间转换为今天昨天前天几天前
- 获取ip接口
- 协程
- 概念
- yield-from && return-values
- 协程与阻塞的思考
- 中间件
- mysqli异步与php的协程
- 代码片段
- pdo 执行的sql语句
- 二进制安全
- 捕捉异常中断
- global
- 利用cookie模拟登陆
- 解析非正常json
- 简单的对称加密算法
- RSA 加密
- 过滤掉emoji表情
- 判断远程图片是否存在
- 一分钟限制请求100次
- 文件处理
- 多文件上传
- 显示所有文件
- 文件下载和上面显示所有文件配合
- 文件的删除,统计,存数组等
- 图片处理
- 简介
- 验证码
- 图片等比缩放
- 批量添加水印
- beanstalkd
- 安装
- 使用
- RabbitMQ
- 简介
- debain安装
- centos安装
- 常用方法
- 入门
- 工作队列
- 订阅,发布
- 路由
- 主题
- 远程调用RPC
- 消息中间件的选型
- .htaccess
- isset、empty、if区别以及0、‘’、null
- php各版本
- php7.2 不向后兼容的改动
- php中的各种坑
- php7改变
- php慢日志
- 邮件
- PHPMailer实现发邮件
- 验证邮件地址真实性
- 文件下载
- FastCgi 与 PHP-fpm 之间的关系
- openssl 加解密
- 反射
- 钩子方法
- 查找插件
- opcode
- opcache使用
- opcache优化
- 分布式一致性hash算法
- 概念
- 哈希算法好坏的四个定义
- php实现
- java实现
- 数组
- jwt
- jwt简介
- 单点登录
- phpize
- GeoIP扩展
- php无法获得https网页内容的解决方案
- homestead运行的脚本
- Unicode和Utf-8转换
- php优化
- kafka
- fpm配置
- configure配置详解