内容: 记录一组为了减少系统调用的次数,采取用于分散读和集中写的函数,readv和writev。
函数原型:
#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。
使用的消息结构体:
struct iovec{
void *iov_base; //starting address of buffer
size_t iov_len; //size of buffer
}
说明:使用时用以上述结构体为储存方式的结构体数组iov[]
writev以顺序iov[0],iov[1]至iov[iovcnt-1]从缓冲区中聚集输出数据。writev返回输出的字节总数。
readv则将读入的数据按照上述同样顺序散布到缓冲区中,readv总先填满一个缓冲区,然后再填写下一个。
readv返回读到的总字节数。如果遇到文件结尾,已无数据可读,则返回0。
使用意义:
1.readv()代表分散读, 即将数据从文件读到分散的内存块中;
writev()代表集中写,即将多块分散的内存一并写入文件中。
2.因为使用read()将数据读到不连续的内存、使用write()将不连续的内存发送出去,要经过多次的
read,write系统调用
3.如果要从文件中读一片连续的数据至进程的不同区域,有两种方案:使用read()一次将它们读至一个较大的
缓冲区中,然后将它们分成若干部分复制到不同的区域;或者调用read()若干次分批将它们读至不同区域。
同样,如果想将程序中不同区域的数据块连续地写至文件,也必须进行类似的处理。
是否可以避免这种多次进行系统调用的开销呢?
答:使用readv()和writev(),它们只需一次系统调用就可以实现在文件和进程的多个缓冲区之间传送数据
免除了多次系统调用或复制数据的开销。
实例:
#include <stdio.h>
#include <sys/uio.h>
#include <fcntl.h>
#include <stdlib.h>
#include <unistd.h>
int main(void)
{
char buf1[13],buf2[13];
struct iovec iov[2];
iov[0].iov_base = buf1;
iov[0].iov_len = 13;
iov[1].iov_base = buf2;
iov[1].iov_len = 13;
int fd = open("test1.txt",O_RDWR);
if(fd < 0){
perror("open error");
return -1;
}
int rsize = readv(fd, iov, 2);
printf("had read = %d\n",rsize);
close(fd);
fd = open("test2.txt", O_RDWR|O_CREAT, S_IRUSR|S_IWUSR);
if(fd < 0){
perror("open error");
return -1;
}
int wsize = writev(fd,iov,2);
printf("had written = %d\n",wsize);
close(fd);
return 0;
}
touch test1.txt并写入26个字母
执行程序后:
查看test2.txt内容验证: