Linux网络编程——tcp并发服务器(poll实现)

想详细彻底地了解poll或看懂下面的代码请参考《Linux网络编程——I/O复用之poll函数》

代码:

[csharp]  view plain  copy
  1. #include   
  2. #include   
  3. #include   
  4. #include   
  5. #include   
  6. #include   
  7. #include   
  8. #include   
  9. #include   
  10. #include   
  11. #include   
  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
  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
  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. }  

运行结果:


你可能感兴趣的:(Linux网络编程)