readv函数将文件描述符读取到分散的内存存快中,分散读,writev把分散的数据一并写入文件描述符中,集中写。
ssize_t readv(int fd,const struct iovec* vector,int count);
ssize_t writev(int fd,const struct iovec* vector,int count);
fd参数是被操作的目标文件描述符。vector参数的类型是iovec结构数组,该结构体描述一块内存区。count参数是vector数组的长度,就是有所少块内存数据需要从fd中读出或者写到fd。readv和writev成功时返回读出/写入的fd的字节数,失败返回-1,并设置errno,简化版的recvmsg和sendmsg。
web服务器上的集中写:
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <fcntl.h>
#define buflen 1024
static const char* status_line[2] = {"200 OK","500 Internal server error"};
int main(int argc,char* argv[])
{
const char* ip = argv[1];
int port = atoi(argv[2]);
const char* file_name = argv[3];
struct sockaddr_in address;
address.sin_family = AF_INET;
address.sin_port =htons(port) ;
inet_pton(AF_INET,ip,&address.sin_addr);
int sock = socket(AF_INET,SOCK_STREAM,0);
int ret = bind(sock,(struct sockaddr*)&address,sizeof(address));
ret = listen(sock,5);
struct sockaddr_in client;
socklen_t client_addrlength = sizeof(client);
int connfd = accept(sock,(struct sockaddr* )&client,&client_addrlength);
char header_buf[buflen];
char * file_buf;
struct stat file_stat;
bool valid = true;
int len =0;
if(stat(file_name,&file_stat)<0){valid = false;}
else{
if(S_ISDIR(file_stat.st_mode)){valid = false;}
else if(file_stat.st_mode& S_IROTH){
int fd = open(file_name,O_RDONLY);
file_buf = new char[file_stat.st_size+1];
memset(file_buf,'0',file_stat.st_size+1);
if(read(fd,file_buf,file_stat.st_size+1)<0){valid = false;}
}
else{
valid = false;
}
if(valid){
ret = snprintf(header_buf,buflen-1,"%s %s
","HTTP/1.1",status_line[1]);
len += ret;
ret = snprintf(header_buf+len,buflen-1-len,"Content-Length:%d
",file_stat.st_size);
len+=ret;
ret = snprintf(header_buf+len,buflen-1-len,"%s","
");
struct iovec iv[2];
iv[0].iov_base = header_buf;
iv[1].iov_len = strlen(header_buf);
iv[2].iov_base = file_buf;
iv[3].iov_len = file_stat.st_size;
ret = writev(connfd,iv,2);
}
else{
ret = snprintf(header_buf,buflen-1,"%s %s
","http/1.1",status_line[1]);
len += ret ;
ret = snprintf(header_buf+len, buflen-1-len,"%s","
");
send(connfd,header_buf,strlen(header_buf),0);
}
close(connfd);
delete[] file_buf;
}
close(sock);
return 0;
}
sendfile在俩个文件描述符之间传递数据(在内核中操作),避免了内核缓冲区和用户缓冲区之间的拷贝,效率很高,零拷贝。
#include<sys/sendfile.h>
ssize_t sendfile(int out_fd,int in_fd,off_t offset,size_t count);
in_fd参数是待读出的文件描述符,out_fd是待写入的文件描述符。offset参数指定读入文件流的那个位置开始读,为空,默认起始位置开始读。count参数指定文件描述符之间的传输的字节数。sendfile成功时候,返回传输的字节数,失败返回-1并设置errno。另外in_fd必须指向真实的文件,而不能是socket或是管道。而out_fd必须是一个socket。由此可见sendfile是专门为网络传输文件而设计的。
例子:
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <fcntl.h>
#include <sys/sendfile.h>
#include <sys/stat.h>
int main(int argc,char* argv[])
{
const char* ip = argv[1];
int port = atoi(argv[2]);
const char* file_name = argv[3];
int filefd = open(file_name,O_RDONLY);
struct stat stat_buf;
fstat(filefd,&stat_buf);
struct sockaddr_in address;
address.sin_family = AF_INET;
address.sin_port = htons(port);
inet_pton(AF_INET,ip,&address.sin_addr);
int sock = socket(PF_INET,SOCK_STREAM,0);
int ret = bind(sock,(struct sockaddr*)&address,sizeof(address));
ret = listen(sock,5);
struct sockaddr_in client;
socklen_t client_addrlength = sizeof(client);
int connfd = accept(sock,(struct sockaddr*)&client,&client_addrlength);
sendfile(connfd,filefd,NULL,stat_buf.st_size);
close(sock);
return 0;
}
6、splice函数。
用于在俩个文件描述符之间移动数据,也是零拷贝操作。splice函数的定义如下:
#include<fcntl.h>
ssize_t splice (int fd_in,loff_t * off_in,int fd_out,loff_t * off_out,size_t len,unsigned int flags);
fd_in 是待输入数据的文件描述符。如果fd_in是一个管道文件描述符,那么off_in参数必须被奢姿为NULL。如果fd_in不似一个管道文件描述符(socket),那么off_in表示从输入数据流的何处开始读取数据。此时,若off_in被设置为null,则表示从输入数据流的当前偏移位置读入;若off_in不为null,则他将指出具体的偏移位置。fd_out/off_out参数的含义与fd_in/off_in相同,不过用于输出数据流。冷参数指定移动数据的长度;
使用splice()时候,fd_in和fd_out必须至少有一个管道文件描述符。splice调用成功时,返回移动字节的数量。
案例:我们使用splice函数实现一个零拷贝的回射服务器,将客户端发送的数据原样返回给客户端。
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <stdio.h>
#include <stdlib.h>
#include <fcntl.h>
#include <string.h>
#include <unistd.h>
#include <iostream>
using namespace std;
int main(int argc,char* argv[]){
const char* ip =argv[1];
int port = atoi(argv[2]);
struct sockaddr_in address;
address.sin_family = AF_INET;
address.sin_port = htons(port);
inet_pton(AF_INET,ip,&address.sin_addr);
int sock= socket(AF_INET,SOCK_STREAM,0);
int ret = bind(sock,(struct sockaddr*)&address,sizeof(address));
ret = listen(sock,5);
struct sockaddr_in client ;
socklen_t client_addrlength = sizeof(client);
int conn = accept(sock,(struct sockaddr*)&client,&client_addrlength);
int pipefd[2];
ret = pipe(pipefd);
ret = splice(pipefd[0],NULL,conn,NULL,32768,SPLICE_F_MORE | SPLICE_F_MOVE);
close(conn);
return 0;
}
7、tee()函数。
tee函数用于在俩个管道文件描述符之间复制数据,也是零拷贝操作。不消耗数据,因此源文件描述符上的数据任然可以用于后续的读操作。tee函数的原型如下;
#include<fcntl.h>
ssize_t tee(int fd_in , int fd_out , size_t len , unsigned int flags);
该函数的参数的含义与splice相同(但fd_in , fd_out 都必须是管道文件描述符)。tee函数成功是返回再俩个文件描述符之间复制的数据流量(字节数)。返回0表示没有复制任何数据。tee失败返回-1,并设施errno。
8、fcntl()函数。
ioctl)比fcntl()更能够执行更多的控制。
#include<fcntl.h>
int fcntl(int fd, int cmd,...);
fd参数是被操作的文件描述符,cmd执行各种类型的操作。根据操作类型的不同,该函数可能还需要第三个可选参数arg。
另外,SIGIO和SIGURG这俩个信号与其他的信号不同,他们必须与某个文件描述符相关联方可使用;