Unix/Linux服务器技术分享自留地PHP经验分享

Nginx+PHP Xsendfile文件传输

2017-06-15  本文已影响79人  Separes

1.简介

传统的文件传输模式中(read/write和send/recv),需要在文件file,系统buffer和用户buffer中反复I/O,造成内存的浪费与资源占用,大致流程如下.

成功返回读取的字节数,出错返回-1并设置errno,如果在调read之前已到达文件末尾,则这次read返回0。
read()会把参数fd所指的文件传送nbyte个字节到buf指针所指的内存中。若参数nbyte为0,则read()不会有作用并返回0。返回值为实际读取到的字节数,如果返回0,表示已到达文件尾或无可读取的数据。错误返回-1,并将根据不同的错误原因适当的设置错误码。

ssize_t write(int fd, const void *buf, size_t nbyte);
write函数把buf中nbyte写入文件描述符handle所指的文档,成功时返回写的字节数,错误时返回-1.

关于Linux User Mode和Kernel Mode

简单图示:


图片来源:http://laoxu.blog.51cto.com/4120547/1417294

这样的传输方式固然简单可靠,但是由于一共进行了四次跨space的I/O和四次mode切换,所以在传输size过大或数量过多的文件时效率堪忧.


在Linux 2.0+以后提供了一个sendfile()的文件传送方式,
ssize_t sendfile(int out_fd, int in_fd, off_t *offset, size_t count);
文档

sendfile()是作用于数据拷贝在两个文件描述符之间的操作函数.这个拷贝操作是内核中操作的,所以称为"零拷贝".sendfile函数比起read和write函数高效得多,因为read和write是要把数据拷贝到用户应用层操作.

其大致流程如下:

以上流程中并没有出现mode的切换,并且省略了涉及user buffer的两次I/O,所以性能会比传统方式优异许多.

简单图示:

图片来源:http://laoxu.blog.51cto.com/4120547/1417294

2.实现

强烈建议先阅读官方文档:
XSendfile-Nginx官方文档

$ sudo vi /etc/nginx/nginx.conf
>>
sendfile on;
header('Content-type: application/octet-stream');
// 这里的$s_fileName指的是被下载的文件名
header('Content-Disposition: attachment; filename="' . $s_fileName . '"');
// nginx sendfile
// 这里的$p_file指的是在nginx中约定的访问路径
header('X-Accel-Redirect: '.$p_file);
// 假设 $p_file = "/demo/download/" . $s_fileName;
// 假设该文件的实际路径为 /var/www/demo/_api.git/var/tmp/
location /demo/download {
    internal;
    alias   /var/www/demo/_api.git/var/tmp/;
}
$ sudo service nginx reload

需要注意的是:

X-Accel-Limit-Rate: 1024
X-Accel-Buffering: yes|no
X-Accel-Charset: utf-8

3.其它:

Apache2 Xsendfile mod
DEMO

上一篇下一篇

猜你喜欢

热点阅读