基本概念
readv和writev函数用于在一次函数调用中读、写多个非连续缓冲区。有时也将这两个函数称为散布读(scatter read)和聚集写(gather write)。
#include<sys/uio.h>
ssize_t readv(int filedes, const struct iovec *iov, int iovcnt);
ssize_t writev(int filedes, const struct iovec *iov, int iovcnt);
//两个函数的返回值:若成功则返回已读、写的字节数,若出错则返回-1
这两个函数的第二个参数是指向iovec结构数组的一个指针:
struct iovec{
void *iov_base; //starting address of buffer
size_t iov_len; //size of buffer
}
iov数组中的元素数由iovcnt说明。下图说明了readv和writev的参数和iovec结构。其最大值受限于IOV_MAX
writev以顺序iov[0],iov[1]至iov[iovcnt-1]从缓冲区中聚集输出数据。writev返回输出的字节总数。
readv则将读入的数据按照上述同样顺序散布到缓冲区中,readv总是先填满一个缓冲区,然后再填写下一个。readv返回读到的总字节数。如果遇到文件结尾,已无数据可读,则返回0。
示例:
#include <stdio.h>
#include <sys/uio.h>
#include <fcntl.h>
#include <unistd.h>
int main(void){
char buf1[5],buf2[10];
struct iovec iov[2];
iov[0].iov_base = buf1;
iov[0].iov_len = 5;
iov[1].iov_base = buf2;
iov[1].iov_len = 10;
int fd = open("a.txt",O_RDWR);
if(fd < 0){
perror("open");
return -1;
}
int rsize = readv(fd, iov, 2);
printf("rsize = %d\n",rsize);
close(fd);
fd = open("b.txt", O_RDWR|O_CREAT, S_IRUSR|S_IWUSR);
if(fd < 0){
perror("open");
return -1;
}
int wsize = writev(fd,iov,2);
printf("wsize = %d\n",wsize);
close(fd);
return 0;
}
注意:需要在程序目录下新建一个a.txt文件,并保存一些数据,如admin:abcd1234
注意事项
- 阻塞IO读写文件/网络socket数据都好使。(会阻塞线程)
- 非阻塞IO读写数据不全时(比如socket缓冲区满),要做大量异常处理和数据重组计算(不太划算)。
- 一定要注意使用这两个函数时,socket是否阻塞、缓冲区大小、返回值处理。
参考资料:
https://blog.csdn.net/todd911/article/details/18792599
https://www.cnblogs.com/nufangrensheng/p/3559304.html