零拷贝操作

零拷贝

在两个文件描述符之间直接传递数据(完全在内核中操作),从而避免了内核缓冲区和用户缓冲区之间的数据拷贝。

接下来,将介绍三个与零拷贝操作相关的函数,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高性能服务器编程》

猜你喜欢

转载自blog.csdn.net/hhhanpan/article/details/79455445
今日推荐