【Linux】readv与writev

来源:互联网 发布:个人信息管理系统源码 编辑:程序博客网 时间:2024/05/20 04:51

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 world12 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 helloreadv11 bytes read.buf1: hellorebuf2: adv
1 0
原创粉丝点击