1.poll的接口介绍
poll系统调用和select类似,也是在指定时间内轮询一定数量的文件描述符,已测试其中是否有就绪者。poll的原型如下:
# include <poll.h> int poll(struct pollfd*fds,nfds_t nfds,int timeout); poll系统调用成功返回就绪文件描述符的总数,超时返回0,失败返回-1 nfds参数指定被监听事件集合fds的大小。 timeout参数指定poll的超时值,单位是毫秒, timeout 为-1 时, poll 调用将永久 阻塞,直到某个事件发生, timeout 为 0 时, poll 调用将立即返回。 fds 参数是一个 struct pollfd 结构类型的数组,它指定所有用户感兴趣的文件描述 符上发生的可读、可写和异常等事件。 pollfd 结构体定义如下: struct pollfd {int fd; // 文件描述符short events; // 注册的关注事件类型short revents; // 实际发生的事件类型,由内核填充 }; 其中, fd 成员指定文件描述符, events 成员告诉 poll 监听 fd 上的哪些事件类型。 它是一系列事件的按位或, revents 成员则有内核修改,通知应用程序 fd 上实际发 生了哪些事件。
事件POLLIN,描述:数据可读
2.epoll的接口介绍
epoll 是 Linux 特有的 I/O 复用函数。它在实现和使用上与 select、 poll 有很大差异。首
先, epoll 使用一组函数来完成任务,而不是单个函数。其次, epoll 把用户关心的文件描述
符上的事件放在内核里的一个事件表中。从而无需像 select 和 poll 那样每次调用都要重复传
入文件描述符或事件集。但 epoll 需要使用一个额外的文件描述符,来唯一标识内核中的这
个事件表。 epoll 相关的函数如下:
epoll_create()用于床建内核事件表epoll_ctl()用于操作内核事件表
epoll_wait()用于在一段超时间内等待一组文件描述符上的事件
# include <sys/epoll.h> int epoll_create(int size); //epoll_create()成功返回内核事件表的文件描述符,失败返回-1 //size 参数现在并不起作用,只是给内核一个提示,告诉它事件表需要多大。 //int epoll_ctl(int epfd,int op,int fd,struct epoll_event*event); epoll_ctl()成功返回 0,失败返回-1 //epfd 参数指定要操作的内核事件表的文件描述符 //fd 参数指定要操作的文件描述符 //op 参数指定操作类型:EPOLL_CTL_ADD 往内核事件表中注册 fd 上的事件EPOLL_CTL_MOD 修改 fd 上的注册事件EPOLL_CTL_DEL 删除 fd 上的注册事件 //event 参数指定事件,它是 epoll_event 结构指针类型 int epoll_wait(int epfd,struct epoll_event*events,int maxevents,int timeout); //epoll_wait()成功返回就绪的文件描述符的个数,失败返回-1,抄实返回0 //epfd 参数指定要操作的内核事件表的文件描述符 //events 参数是一个用户数组,这个数组仅仅在 epoll_wait 返回时保存内核检测到 的所有就绪事件,而不像 select 和 poll 的数组参数那样既用于传入用户注册的事 件,又用于输出内核检测到的就绪事件。这就极大地提高了应用程序索引就绪文件 描述符的效率。 //maxevents 参数指定用户数组的大小,即指定最多监听多少个事件,它必须大于 0 //timeout 参数指定超时时间,单位为毫秒,如果 timeout 为 0,则 epoll_wait 会立即 返回,如果 timeout 为-1,则 epoll_wait 会一直阻塞,直到有事件就绪。
2.1 LT和ET模式
epoll对文件描述符有两种操作模式:LT模式和ET模式。LT模式是默认工作模式。当epoll内核事件表中注册一个文件描述符上的EPOLLET事件时,epoll将以高效的ET模式来操作该文件描述符。
对于LT模式操作的文件描述符,当epoll_wait检测到其上有事物发生并将此事通知应用程序后,应用程序可以不立即处理该事物。这样,当应用程序下一次调用epoll_wait时,还会再次向应用程序通告此事件,直到该事件被处理。
对于ET模式操作的文件描述符,当epoll_wait检测到其上有事件,并将此事件通知应用程序后,应用程序必须立即处理该事件,因为后续的epoll_wait调用将不再向应用程序通知这一事件。所以ET模式在很大程度上降低了同一个epoll事件被重复触发的次数,因此效率该与LT模式。
epoll实现tcp服务器代码如下:
设置文件为非阻塞模式
void SetNoWait(int fd) {int old_option=fcntl(fd,F_GETFL);int new_option=old_option|O_NONBLOCK;fcntl(fd,F_SETFL,new_option); }
关闭客户端连接
void CloseClient(int epfd, int fd) {close(fd);if (epoll_ctl(epfd, EPOLL_CTL_DEL, fd, NULL) == -1){printf("epoll_ctl del error\n");} }
获取一个新的客户端连接,如果 flag 为 ET,则以 ET 模式处理此客户端
void GetClientLink(int sockfd, int epfd, int flag) {struct sockaddr_in caddr;socklen_t len = sizeof(caddr);int c = accept(sockfd, (struct sockaddr*)&caddr, &len);if (c < 0){printf("Client Link error\n");return;}struct epoll_event ev;ev.data.fd = c;if (flag){ev.events = EPOLLIN | EPOLLRDHUP | EPOLLET;SetNoWait(c);}else{ev.events = EPOLLIN | EPOLLRDHUP;}if (epoll_ctl(epfd, EPOLL_CTL_ADD, c, &ev) == -1){printf("epoll_ctl add error\n");}} }