《Linux 网络编程五:TCP 并发服务器:构建模式、原理及关键技术(epoll )》
本文对比了三种并发服务器的实现方式(select、poll、epoll),重点分析了epoll的工作原理和优势。select和poll都存在文件描述符数量限制、数据拷贝开销大、需要遍历事件等缺点,而epoll采用红黑树结构,支持高效查找,避免了数据拷贝,并能直接返回就绪事件。epoll支持水平触发和边沿触发两种模式,通过epoll_create、epoll_ctl和epoll_wait三个核心函数
一、三种并发服务器对比
1.select特点:
1.使用位图(数组)实现对文件描述符集合的保存,最多允许同时监测1024个文件描述符。
2.需要应用和内核层的反复数据(文件描述符集合表)拷贝
3.返回的集合表需要遍历寻找到达的事件
4.只能工作在水平触发模式(低速模式),不能工作在边沿触发模式(高速模式)
2.poll特点:
1.使用链表实现对文件描述符集合的保存,没有监测文件描述符个数的上限。
2.需要应用和内核层的反复数据(文件描述符集合表)拷贝
3.返回的集合表需要遍历寻找到达的事件
4.只能工作在水平触发模式(低速模式),不能工作在边沿触发模式(高速模式)
3.epoll特点:
1.使用红黑树(二叉树)实现对文件描述符集合的保存,没有监测文件描述符个数的上限,提高查找效率。
2.将文件描述符集合创建在内核层,避免了应用层和内核层的数据反复拷贝。
3.返回到达的事件,不需要遍历,只需要处理事件即可。
4.可工作在水平触发模式(低速模式)和边沿触发模式(高速模式)
水平触发模式(低速模式):高低电平触发事件
边沿触发模式(高速模式):状态变化时触发事件(上升沿触发,下降沿触发)
二、epoll工作流程
epoll:
1.创建文件描述符集合,epoll_create
2.添加关注的文件描述符,epoll_ctl
3.epoll通知内核开始事件监测,epoll_wait
4.epoll返回时,获取到达事件的结果
5.根据到达事件做任务处理
#include <sys/epoll.h>
int epoll_create(int size);
功能:通知内核创建文件描述符集合
参数:
size:监测的文件描述符个数
返回值:
成功:文件描述符(内核创建的集合)
失败:-1
#include <sys/epoll.h>
int epoll_ctl(int epfd, int op, int fd, struct epoll_event *event);
功能:对epoll的文件描述符集合进行操作
参数:
epfd:创建的epoll集合
op:对文件描述符集合进行的操作
EPOLL_CTL_ADD:添加集合中的文件描述符
EPOLL_CTL_MOD:修改集合中的文件描述符
EPOLL_CTL_DEL:删除集合中的文件描述符
fd:要操作的文件描述符
event:文件描述符对应的事件
typedef union epoll_data {
void *ptr;
int fd;
uint32_t u32;
uint64_t u64;
} epoll_data_t;
struct epoll_event {
uint32_t events; /* Epoll events */
epoll_data_t data; /* User data variable */
};
events:
EPOLLIN:读事件
EPOLLOUT:写事件
data.fd关注的描述符事件
返回值:
成功:0
失败:-1
#include <sys/epoll.h>
int epoll_wait(int epfd, struct epoll_event *events,
int maxevents, int timeout);
功能:通知内核开始监测文件描述符的事件
参数:
epfd:监测的文件描述符集合
events:保存返回的到达事件的结果
struct epoll_event evs[2]; 传数组名(evs)->可以保存多个
maxevents:最大的事件个数
timeout:监测的超时时间;-1不设置超时(阻塞)
返回值:
成功:到达的事件个数
失败:-1
三、相关代码
1.用epoll构建并发服务器
#ifndef __HEAD_H__
#define __HEAD_H__
#include <stdio.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdlib.h>
#include <string.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netinet/ip.h> /* superset of previous */
#include <arpa/inet.h>
#include <pthread.h>
#include <sys/select.h>
/* According to earlier standards */
#include <sys/time.h>
#include <sys/types.h>
#include <unistd.h>
#include <sys/epoll.h>
#endif
#include "head.h"
#define MAX_FD_CNT 100
int init_tcp_ser()
{
int sockfd = socket(AF_INET, SOCK_STREAM, 0);
if(sockfd < 0)
{
perror("socket error");
return -1;
}
//允许绑定处于TIME_WAIT状态的地址,避免端口占用问题:
int optval = 1;
setsockopt(sockfd, SOL_SOCKET, SO_REUSEADDR, &optval, sizeof(optval));
struct sockaddr_in seraddr;
seraddr.sin_family = AF_INET;
seraddr.sin_port = htons(50001);
seraddr.sin_addr.s_addr = inet_addr("192.168.0.192");
int ret = bind(sockfd,(struct sockaddr *)&seraddr, sizeof(seraddr));
if(ret < 0)
{
perror("bind error");
return -1;
}
ret = listen(sockfd, 10);
if(ret < 0)
{
perror("listen error");
return -1;
}
return sockfd;
}
int epoll_fd_add(int epfds, int fd, uint32_t events)
{
struct epoll_event ev;
ev.events = events;
ev.data.fd = fd;
int ret = epoll_ctl(epfds, EPOLL_CTL_ADD, fd, &ev);
if(ret < 0)
{
perror("epoll_ctl error");
return -1;
}
return 0;
}
int epoll_fd_del(int epfds, int fd, uint32_t events)
{
int ret = epoll_ctl(epfds, EPOLL_CTL_DEL, fd, NULL);
if(ret < 0)
{
perror("epoll_ctl error");
return -1;
}
return 0;
}
int main(int argc, char const *argv[])
{
int sockfd = init_tcp_ser();
char buff[1024] = {0};
struct sockaddr_in cliaddr;
socklen_t clilen = sizeof(cliaddr);
int epfds = epoll_create(MAX_FD_CNT);
if(epfds < 0)
{
perror("epoll_creat error");
return -1;
}
epoll_fd_add(epfds, sockfd, EPOLLIN);
struct epoll_event evs[MAX_FD_CNT];
while(1)
{
int cnt = epoll_wait(epfds, evs, MAX_FD_CNT, -1);
if(cnt < 0)
{
perror("epoll_wait error");
return -1;
}
for(int i = 0; i < cnt; ++i)
{
int fd = evs[i].data.fd;
if(sockfd == fd)
{
int connfd = accept(sockfd, (struct sockaddr *)&cliaddr, &clilen);
epoll_fd_add(epfds, connfd, EPOLLIN);
}
else
{
memset(buff, 0, sizeof(buff));
ssize_t cnt = recv(fd, buff, sizeof(buff), 0);
if (cnt < 0)
{
perror("recv error");
epoll_fd_del(epfds, fd, EPOLLIN);
close(fd);
continue;
}
else if (0 == cnt)
{
epoll_fd_del(epfds, fd, EPOLLIN);
close(fd);
continue;
}
printf("%s\n", buff);
strcat(buff, "--->ok");
cnt = send(fd, buff, strlen(buff), 0);
if (cnt < 0)
{
perror("send error");
epoll_fd_del(epfds, fd, EPOLLIN);
close(fd);
continue;
}
}
}
}
return 0;
}更多推荐


所有评论(0)