想详细彻底地了解poll或看懂下面的代码请参考《Linux网络编程——I/O复用之poll函数》
1 #include <string.h> 2 #include <stdio.h> 3 #include <stdlib.h> 4 #include <unistd.h> 5 #include <sys/select.h> 6 #include <sys/time.h> 7 #include <sys/socket.h> 8 #include <netinet/in.h> 9 #include <arpa/inet.h> 10 #include <poll.h> 11 #include <errno.h> 12 #define OPEN_MAX 100 13 14 int main(int argc, char *argv[]) 15 { 16 //1.创建tcp监听套接字 17 int sockfd = socket(AF_INET, SOCK_STREAM, 0); 18 19 //2.绑定sockfd 20 struct sockaddr_in my_addr; 21 bzero(&my_addr, sizeof(my_addr)); 22 my_addr.sin_family = AF_INET; 23 my_addr.sin_port = htons(8000); 24 my_addr.sin_addr.s_addr = htonl(INADDR_ANY); 25 bind(sockfd, (struct sockaddr *)&my_addr, sizeof(my_addr)); 26 27 //3.监听listen 28 listen(sockfd, 10); 29 30 //4.poll相应参数准备 31 struct pollfd client[OPEN_MAX]; 32 int i = 0, maxi = 0; 33 for(;i<OPEN_MAX; i++) 34 client[i].fd = -1;//初始化poll结构中的文件描述符fd 35 36 client[0].fd = sockfd;//需要监测的描述符 37 client[0].events = POLLIN;//普通或优先级带数据可读 38 39 //5.对已连接的客户端的数据处理 40 while(1) 41 { 42 int ret = poll(client, maxi+1, -1);//对加入poll结构体数组所有元素进行监测 43 44 //5.1监测sockfd(监听套接字)是否存在连接 45 if((client[0].revents & POLLIN) == POLLIN ) 46 { 47 struct sockaddr_in cli_addr; 48 int clilen = sizeof(cli_addr); 49 int connfd = 0; 50 //5.1.1 从tcp完成连接中提取客户端 51 connfd = accept(sockfd, (struct sockaddr *)&cli_addr, &clilen); 52 53 //5.1.2 将提取到的connfd放入poll结构体数组中,以便于poll函数监测 54 for(i=1; i<OPEN_MAX; i++) 55 { 56 if(client[i].fd < 0) 57 { 58 client[i].fd = connfd; 59 client[i].events = POLLIN; 60 break; 61 } 62 } 63 64 //5.1.3 maxi更新 65 if(i > maxi) 66 maxi = i; 67 68 //5.1.4 如果没有就绪的描述符,就继续poll监测,否则继续向下看 69 if(--ret <= 0) 70 continue; 71 } 72 73 //5.2继续响应就绪的描述符 74 for(i=1; i<=maxi; i++) 75 { 76 if(client[i].fd < 0) 77 continue; 78 79 if(client[i].revents & (POLLIN | POLLERR)) 80 { 81 int len = 0; 82 char buf[128] = ""; 83 84 //5.2.1接受客户端数据 85 if((len = recv(client[i].fd, buf, sizeof(buf), 0)) < 0) 86 { 87 if(errno == ECONNRESET)//tcp连接超时、RST 88 { 89 close(client[i].fd); 90 client[i].fd = -1; 91 } 92 else 93 perror("read error:"); 94 95 } 96 else if(len == 0)//客户端关闭连接 97 { 98 close(client[i].fd); 99 client[i].fd = -1; 100 } 101 else//正常接收到服务器的数据 102 send(client[i].fd, buf, len, 0); 103 104 //5.2.2所有的就绪描述符处理完了,就退出当前的for循环,继续poll监测 105 if(--ret <= 0) 106 break; 107 108 } 109 } 110 } 111 return 0; 112 }
原文:https://www.cnblogs.com/jiangzhaowei/p/8831079.html