零拷贝
在两个文件描述符之间直接传递数据(完全在内核中操作),从而避免了内核缓冲区和用户缓冲区之间的数据拷贝。
接下来,将介绍三个与零拷贝操作相关的函数,sendfile函数,splice函数和tee函数
首先,
sendfile函数
#include 《sys/sendfile.h>
ssize_t sendfile(int out_fd, int in_fd, off_t* offset, size_t count );
in_fd参数是待读出内容的文件描述符(in_fd必须是一个支持类似mmap函数描述符,即指向
真实的文件,不能是socket或管道)。
out_fd是待写入内容的文件描述符(out_fd必须是一个socket)。
offset是指定读入文件流的哪个位置开始读,如果为空,则默认的起始位置。
count参数指定在文件描述in_fd和out_fd之间传递的字节数。
函数成功返回传输的字节数;失败返回-1并设置error.
splice函数
用于在两个文件描述符之间移动数据,也是零拷贝操作。
#include <fcntl.h>
ssize_t splice( int fd_in, loff_t* off_in, int fd_out, loff_t* ff_out,
size_t len, unsigned int flags );
fd_out/off_out参数与fd_in/offin类似,用于输出数据流。
len参数指定移动数据的长度。
flag参数用于控制数据如何移动。
函数调用成功时返回移动字节的数量,返回0表示没有数据需要移动(发生在从管道中读取数据
而该管道没有被写入任何数据)。
失败返回-1并设置error。
tee函数
该函数是在两个管道文件描述符之间复制数据。不消耗数据(源文件描述符上的数据仍可以用于后续的读操作)
#include <fcntl.h>
ssize_t tee ( int fd_in, int fd_out, size_t len, unsigned int flags );
这里的fd_in和fd_out必须是管道文件描述符
函数成功返回在两个文件描述符之间复制的数据数量(字节数),0表示没有复制任何数据,
失败返回-1并设置errno.
参考资料:《Linux高性能服务器编程》