本文转自【 http://blog.csdn.net/ieearth/article/details/46730669 】
readv和writev函数是Linux中的两个系统调用,类似于read和write函数,不同的是,readv和writev在一次执行过程中可以原子地作用于多个缓冲区,这些缓冲区常常是非连续的。readv和writev的原型如下:
#include <sys/uio.h> ssize_t readv(int fd, const struct iovec *iov, int iovcnt); ssize_t writev(int fd, const struct iovec *iov, int iovcnt); struct iovec { void *iov_base; /* Starting address */ size_t iov_len; /* Number of bytes to transfer */ };
readv和writev的第一个参数fd是个文件描述符,第二个参数是指向iovec数据结构的一个指针,其中iov_base为缓冲区首地址,iov_len为缓冲区长度,参数iovcnt指定了iovec的个数。函数调用成功时返回读、写的总字节数,失败时返回-1并设置相应的errno。
在一次函数调用中,writev以顺序iov[0]、iov[1]至iov[iovcnt-1]从各缓冲区中聚集输出数据到fd,readv则将从fd读入的数据按同样的顺序散布到各缓冲区中,readv总是先填满一个缓冲区,然后再填下一个,因此,writev称为gather output,readv称为scatter input。
先来看一个writev的例子,指定了两个缓冲区,str0和str1,内容输出到标准输出,并打印实际输出的字节数。
// writevex.c #include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <string.h> #include <sys/uio.h> int main() { char *str0 = "hello "; char *str1 = "world\n"; struct iovec iov[2]; ssize_t nwritten; iov[0].iov_base = str0; iov[0].iov_len = strlen(str0); iov[1].iov_base = str1; iov[1].iov_len = strlen(str1); nwritten = writev(STDOUT_FILENO, iov, 2); printf("%ld bytes written.\n", nwritten); exit(EXIT_SUCCESS); }
$ gcc writevex.c $ ./a.out hello world 12 bytes written.
再来看一个readv的例子,从标准输入读数据,缓冲区为长度是(8 – 1)的buf1和buf2,并打印读到的字节总数和两个缓冲区各自的内容。
// readvex.c #include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <string.h> #include <sys/uio.h> int main() { char buf1[8] = { 0 }; char buf2[8] = { 0 }; struct iovec iov[2]; ssize_t nread; iov[0].iov_base = buf1; iov[0].iov_len = sizeof(buf1) - 1; iov[1].iov_base = buf2; iov[1].iov_len = sizeof(buf2) - 1; nread = readv(STDIN_FILENO, iov, 2); printf("%ld bytes read.\n", nread); printf("buf1: %s\n", buf1); printf("buf2: %s\n", buf2); exit(EXIT_SUCCESS); }
$ gcc readvex.c $ ./a.out helloreadv 11 bytes read. buf1: hellore buf2: adv
发表评论