sendfile函数
Posted qq_34132502
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了sendfile函数相关的知识,希望对你有一定的参考价值。
sendfile
函数在两个文件描述符中间直接传递数据(完全在内核中操作),从而避免了内核缓冲区和用户缓冲区之间的数据拷贝,效率很高,这被称为零拷贝。
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参数指定传输的字节数。
其中in_fd必须指向真是文件,不能使socket或管道。而out_fd则必须为socket
用sendfile
函数传输文件:
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <assert.h>
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <errno.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <sys/sendfile.h>
#include <libgen.h>
#define BACKLOG 5
int main(int argc, char* argv[]) {
if (argc <= 3) {
printf("usage: %s ip_address port_number filename\\n", basename(argv[0]));
return 1;
}
const char* ip = argv[1];
int port = atoi(argv[2]);
const char* file_name = argv[3];
int filefd = open(file_name, O_RDONLY);
assert(filefd > 0);
struct stat stat_buf;
fstat(filefd, &stat_buf);
struct sockaddr_in address;
bzero(&address, sizeof(address));
inet_pton(AF_INET, ip, &address.sin_addr);
address.sin_port = htons(port);
address.sin_family = AF_INET;
int sock = socket(AF_INET, SOCK_STREAM, 0);
assert(sock >= 0);
int ret = bind(sock, (struct sockaddr*)&address, sizeof(address));
assert(ret != -1);
ret = listen(sock, BACKLOG);
assert(ret != -1);
struct sockaddr_in client;
socklen_t client_addrlength = sizeof(client);
int connfd = accept(sock, (struct sockaddr*)&client, &client_addrlength);
if (connfd < 0) {
printf("errno is %d\\n", errno);
} else {
sendfile(connfd, filefd, NULL, stat_buf.st_size);
close(connfd);
}
close(sock);
return 0;
}
我们将目标文件作为参数传送给服务器程序,客户telnet到该可武器上即可获取该文件。
此例中,没有给目标文件分配任何缓存,也没有读取文件的操作,比之前发送文件的代码要简单很多,效率也更高。
以上是关于sendfile函数的主要内容,如果未能解决你的问题,请参考以下文章