Linux后台网络编程中select/poll/epoll的比较分析
一.select
1.概述
select本质是通过设置或检查存放fd标志位的数据结构来进行下一步的处理。会阻塞,直到有一个或多个I/O就绪。
监视的文件描述符分为三类set,每一种对应不同的事件。readfds、writefds和exceptfds是指向描述符集的指针。
readfds列出的文件描述符被监视是否有数据可供读取。(可读)
writefds列出的文件描述符被监视是否有写入操作完成。(可写)
exceptfds列出的文件描述符被监视是否发生异常,或无法控制的数据是否可用。(仅仅用于socket)
这三类set为NULL时,select()不监视其对应的该类事件。
select()成功返回时,每组set都被修改以使它只包含准备好的I/O描述符。
其缺点:(a)单个进程可监视的fd数量被限制;
(b)需要维护一个用来存放大量fd的数据结构,这样会使用户空间和内核空间在传递该结构时复制开销大;
(c)对fd进行扫描是线性的,fd剧增后,IO效率较低,因为每次调用都对fd进行线性扫描遍历,所以随着fd的增加会造成遍历速度慢的性能问题;
(d)内核需要将消息传递用户空间,需要内核拷贝动作;
(e)最大支持1024个fd。
2.实战
server端代码,文件名为:select-server.c
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#define MAXBUF 1024
/************关于本文档********************************************
*filename: select-server.c
*purpose: 演示网络异步通讯、select用法,这是服务器端程序
*Note: 任何人可以任意复制代码并运用这些文档,当然包括你的商业用途
* 但请遵循GPL
*Thanks to: Google.com
*Hope:希望越来越多的人贡献自己的力量,为科学技术发展出力
* 科技站在巨人的肩膀上进步更快!感谢有开源前辈的贡献!
*********************************************************************/
int main(int argc, char **argv)
{
int sockfd, new_fd;
socklen_t len;
struct sockaddr_in my_addr, their_addr;
unsigned int myport, lisnum;
char buf[MAXBUF + 1];
fd_set rfds;
struct timeval tv;
int retval, maxfd = -1;
if (argv[1])
myport = atoi(argv[1]);
else
myport = 7838;
if (argv[2])
lisnum = atoi(argv[2]);
else
lisnum = 2;
if ((sockfd = socket(PF_INET, SOCK_STREAM, 0)) == -1) {
perror(“socket”);
exit(1);
}
bzero(&my_addr, sizeof(my_addr));
my_addr.sin_family = PF_INET;
my_addr.sin_port = htons(myport);
if (argv[3])
my_addr.sin_addr.s_addr = inet_addr(argv[3]);
else
my_addr.sin_addr.s_addr = INADDR_ANY;
if (bind(sockfd, (struct sockaddr *) &my_addr, sizeof(struct sockaddr))
== -1) {
perror(“bind”);
exit(1);
}
if (listen(sockfd, lisnum) == -1) {
perror(“listen”);
exit(1);
}
while (1) {
printf
(“\n—-等待新的连接到来开始新一轮聊天……\n”);
len = sizeof(struct sockaddr);
if ((new_fd =
accept(sockfd, (struct sockaddr *) &their_addr,
&len)) == -1) {
perror(“accept”);
exit(errno);
} else
printf(“server: got connection from %s, port %d, socket %d\n”,
inet_ntoa(their_addr.sin_addr),
ntohs(their_addr.sin_port), new_fd);
/* 开始处理每个新连接上的数据收发 */
printf
(“\n准备就绪,可以开始聊天了……直接输入消息回车即可发信息给对方\n”);
while (1) {
/* 把集合清空 */
FD_ZERO(&rfds);
/* 把标准输入(stdin)句柄0加入到集合中 */
FD_SET(0, &rfds);
maxfd = 0;
/* 把当前连接(socket)句柄new_fd加入到集合中 */
FD_SET(new_fd, &rfds);
if (new_fd > maxfd)
maxfd = new_fd;
/* 设置最大等待时间 */
tv.tv_sec = 1;
tv.tv_usec = 0;
/* 开始等待 */
retval = select(maxfd + 1, &rfds, NULL, NULL, &tv);
if (retval == -1) {
printf(“将退出,select出错! %s”, strerror(errno));
break;
} else if (retval == 0) {
/* printf
(“没有任何消息到来,用户也没有按键,继续等待……\n”); */
continue;
} else {
/*判断当前IO是否是stdin*/
if (FD_ISSET(0, &rfds)) {
/* 用户按键了,则读取用户输入的内容发送出去 */
bzero(buf, MAXBUF + 1);
fgets(buf, MAXBUF, stdin);
if (!strncasecmp(buf, “quit”, 4)) {
printf(“自己请求终止聊天!\n”);
break;
}
len = send(new_fd, buf, strlen(buf) – 1, 0);
if (len > 0)
printf
(“消息:%s\t发送成功,共发送了%d个字节!\n”,
buf, len);
else {
printf
(“消息’%s’发送失败!错误代码是%d,错误信息是’%s’\n”,
buf, errno, strerror(errno));
break;
}
}
/*判断当前IO是否是来自socket*/
if (FD_ISSET(new_fd, &rfds)) {
/* 当前连接的socket上有消息到来则接收对方发过来的消息并显示 */
bzero(buf, MAXBUF + 1);
/* 接收客户端的消息 */
len = recv(new_fd, buf, MAXBUF, 0);
if (len > 0)
printf
(“接收消息成功:’%s’,共%d个字节的数据\n”,
buf, len);
else {
if (len < 0)
printf
(“消息接收失败!错误代码是%d,错误信息是’%s’\n”,
errno, strerror(errno));
else
printf(“对方退出了,聊天终止\n”);
break;
}
}
}
}
close(new_fd);
/* 处理每个新连接上的数据收发结束 */
printf(“还要和其它连接聊天吗?(no->退出)”);
fflush(stdout);
bzero(buf, MAXBUF + 1);
fgets(buf, MAXBUF, stdin);
if (!strncasecmp(buf, “no”, 2)) {