资讯专栏INFORMATION COLUMN

PHP下载文件、限速、X-sendfile

Lionad-Morotar / 1540人阅读

摘要:主要用于存储速度不同步的设备或者优先级不同的设备之间传办理数据的区域。机制,意味在之前,建立了一新的队列,数据必须经过该队列。当需要下载的文件很大的时候,这种方式将消耗大量内存,甚至引发进程超时或崩溃,接下来就使用到。

一、普通文件下载

①laravel框架HTTP响应的download方法

$pathToFile = "myfile.csv";//参数一:绝对路径
$downloadName = "downloadFile.csv";//参数二:下载后的文件名
//download 参数三:HTTP头信息
return response()->download($pathToFile, $downloadName);

②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/10...
但是有时候为了节省带宽,避免瞬时流量过大而造成网络堵塞,就要考虑下载限速的问题

二、下载文件限速
  $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/superhos...

在没有开启缓存时,脚本输出的内容都在服务器端处于等待输出的状态,flush()可以将等待输出的内容立即发送到客户端。

开启缓存后,脚本输出的内容存入了输出缓存中,这时没有处于等待输出状态的内容,你直接使用flush()不会向客户端发出任何内容。而ob_flush()的作用就是将本来存在输出缓存中的内容取出来,设置为等待输出状态,但不会直接发送到客户端,这时你就需要先使用ob_flush()再使用flush(),客户端才能立即获得脚本的输出。

以及这篇文章同样讲述了ob_flush()和flush()的区别http://www.laruence.com/2010/...

但是这种方法将文件内容从磁盘经过一个固定的 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 . "不存在");
}

如果你还想了解更多关于X-sendfile,请自行查阅

记得关注我呦

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/26165.html

相关文章

  • Yii 的修行之路 - HTTP的请求与响应

    摘要:简述交互主要分为请求和响应两种方式。状态码构建响应时,最先应做的是标识请求是否成功处理的状态,可通过设置属性,该属性使用一个有效的状态码。 简述 HTTP交互主要分为 request(请求) 和 response(响应) 两种方式。 对于Yii2来说,HTTP的request请求是用 YII::$app->request 来表示的对象,这是Yii提供的处理HTTP的request请求的...

    import. 评论0 收藏0
  • nginx限速不得不说的事之连接数限制技巧

    摘要:但是,你的连接数限制配置为允许单个连接数,单个连接数最大带宽为。就降低单个连接数带宽吧要知道大家谁没事会用浏览器自带下载器下载呢注本文只探讨限速模块在不同业务下的限速彩蛋偶尔发现,将连接数限制为迅雷不能高速下载了。 nginx 内置模块限速怎么使用就不多说了,今天来说说连接数和单个连接数限速的事。 场景:A公司有100人,A公司只有一个公网IP,假设A公司可能有100个人同时在下载你的...

    neroneroffy 评论0 收藏0
  • nginx限速不得不说的事之连接数限制技巧

    摘要:但是,你的连接数限制配置为允许单个连接数,单个连接数最大带宽为。就降低单个连接数带宽吧要知道大家谁没事会用浏览器自带下载器下载呢注本文只探讨限速模块在不同业务下的限速彩蛋偶尔发现,将连接数限制为迅雷不能高速下载了。 nginx 内置模块限速怎么使用就不多说了,今天来说说连接数和单个连接数限速的事。 场景:A公司有100人,A公司只有一个公网IP,假设A公司可能有100个人同时在下载你的...

    remcarpediem 评论0 收藏0
  • 自己做了一个P2P网盘加速器,基本不限速!——ENFI下载

    摘要:对于那些不想挂机还想使用高速下载的用户来说,只要付费即可。元,元,元,满足你的不同下载需求,如果你是周末偶尔看看电影,偶尔下载点学习资料,那么元的流量包最合适不过。 pandownload会被限速,这众所周知,先看下我这个网盘的下载速度吧showImg(https://segmentfault.com/img/bVbvc7I); 下面我说说为啥我这个支持免登陆下载,而且不受到百度网盘的...

    Shonim 评论0 收藏0
  • 自己做了一个P2P网盘加速器,基本不限速!——ENFI下载

    摘要:对于那些不想挂机还想使用高速下载的用户来说,只要付费即可。元,元,元,满足你的不同下载需求,如果你是周末偶尔看看电影,偶尔下载点学习资料,那么元的流量包最合适不过。 pandownload会被限速,这众所周知,先看下我这个网盘的下载速度吧showImg(https://segmentfault.com/img/bVbvc7I); 下面我说说为啥我这个支持免登陆下载,而且不受到百度网盘的...

    W_BinaryTree 评论0 收藏0

发表评论

0条评论

最新活动
阅读需要支付1元查看
<