nginx代码非常的优秀,之前已经介绍了sendfile的使用来实现“内存零拷贝”,今天我又理解了一下writev函数的使用,以nginx-1.6.0为例,在src/os/unix/ngx_writev_chain.c的113行,如下图:

wKiom1PNwbzhiKizAADnGk8J1EI010.jpg

可以man 2 writev看下函数的概念(与writev相对应的还有一个readv)writev是读取多个不连续的buffer然后集中写入。

#include <sys/uio.h>
      ssize_t readv(int fd, const struct iovec *vector, int count);
      ssize_t writev(int fd, const struct iovec *vector, int count);

如何实现

#include <stdio.h>
#include <fcntl.h>
#include <sys/uio.h>
int main(int argc,char *argv[])
{
        int fd1,fd2,fd3;
        ssize_t size;
        char buf1[9],buf2[9];
        struct iovec iov[2];
     
        fd1=open(argv[1],O_RDONLY);
        fd2=open(argv[2],O_RDONLY);
        fd3=open(argv[3],O_RDWR);
     
        size=read(fd1,buf1,sizeof(buf1));
        printf("%s size is : %d\n",argv[1],size);
     
        size=read(fd2,buf2,sizeof(buf2));
        printf("%s size is : %d\n",argv[2],size);
     
        iov[0].iov_base=buf1;
        iov[0].iov_len=sizeof(buf1);
        iov[1].iov_base=buf2;
        iov[1].iov_len=sizeof(buf2);
        size=writev(fd3,iov,2);
        printf("%s size is :%d\n",argv[3],size);
        close(fd1);
        close(fd2);
        close(fd3);
}

首先创建3个文件,file1(9个字节),file2(9个字节),file3(空文件),程序将file1和file1读入buffer然后将两段buffer集中写入新的fd中

wKioL1PNymWzw6FTAAArEsGy7aE932.jpg

编译测试吧

wKioL1PNy9OyChxcAABHa_3j1_E063.jpg

writev和write函数区别就是在于多个非连续buffer的读取后写入,当负载大的时候就可以很好的体现出性能效果了。