我在之前一篇笔记里面写了如何用php下载文件,里面写了两种方法来下载文件

  • nginx配置直接try_files下载文件
  • php读取文件之后再输出

对于第二种方法,php-fpm和nginx的内存和IO情况都有飙升。


在传统的文件传输过程中,文件数据经历了以下步骤:

硬盘 -> 内核buffer -> 用户buffer -> socket相关缓冲区 -> 协议栈

而sendfile机制被开启之后,减少了一次文件copy,提升了文件传输性能,文件数据经历的步骤如下:

硬盘 -> 内核buffer -> socket相关缓冲区 -> 协议栈

在当前流行的几个web服务器(apache、nginx、lighttpd等)里面,都有关于sendfile的配置。


下面说说如何使用nginx的XSendfile来提升PHP文件下载性能。

关于nginx的XSendfile的详细内容可以看官方文档

1. nginx开启sendfile

在nginx.conf里面,设置sendfile on表示开启sendfile机制

2. 配置路由规则

location /protectedFile/ {
  internal;
  root   /home/nemo/fileDir;
}

internal表示这个路径只能在nginx内部访问,不能在浏览器里面被访问。

配置完之后,重启nginx。

3. 修改php代码

<?php

	
    checkAuthority(); // 做一些业务相关的用户鉴权相关的操作
	……
	……
	 
    $filePath = '/protectedFile/test.pdf';
    header('Content-type: application/octet-stream');
    header('Content-Disposition: attachment; filename="' . basename($filePath) . '"');
    // 让Xsendfile发送文件
    header('X-Accel-Redirect: '.$filePath);
?>

在php代码里面使用X-Accel-Redirect让nginx使用XSendfile机制提升下载性能,这样用户下载到了/home/nemo/fileDir/protectedFile/test.pdf文件

比较一下使用与不使用XSendfile,下载一份20MB+的文件的时候的性能比较

性能指标 不使用XSendfile 使用XSendfile
耗时 1.13s 456.32ms
内存 26.1MB 2.6MB

参考

1.扯淡nginx的sendfile零拷贝的概念