【网络编程】poll函数

立山  金牌会员 | 2024-7-15 21:48:15 | 显示全部楼层 | 阅读模式
打印 上一主题 下一主题

主题 580|帖子 580|积分 1740

poll和select的区别不大,主要是poll没有连接数限定,因为它用的链表实现
  1. #include <poll.h>
  2. int poll(struct pollfd *fds, nfds_t nfds, int timeout);
  3. struct pollfd {
  4.     int   fd;         //要监控的文件描述符,如果fd为-1, 表示内核不再监控
  5.     short events;     //输入参数, 表示告诉内核要监控的事件, 读事件, 写事件, 异常事件
  6.     short revents;    //输出参数, 表示内核告诉应用程序有哪些文件描述符有事件发生
  7. };
  8. events/revents:
  9.                    POLLIN:可读事件
  10.                    POLLOUT: 可写事件
  11.                    POLLERR: 异常事件
  12. nfds: 告诉内核监控的范围, 具体是: 数组下标的最大值+1
  13. timeout:
  14.         =0: 不阻塞, 立刻返回
  15.         -1: 表示一直阻塞, 直到有事件发生
  16.         >0: 表示阻塞时长, 在时长范围内若有事件发生会立刻返回;
  17.                 如果超过了时长也会立刻返回
  18. 函数返回值:
  19.         >0: 发生变化的文件描述符的个数
  20.         =0: 没有文件描述符发生变化
  21.         -1: 表示异常
复制代码
使用poll来监控多个文件描述符举行客户端通讯
代码:
  1. #include "socketwrap.h"
  2. #include <arpa/inet.h>
  3. #include <poll.h>
  4. #include <strings.h>
  5. #include <string.h>
  6. #include <ctype.h>
  7. #define POLLLEN 1024
  8. int main()
  9. {
  10.     int sfd = Socket(AF_INET, SOCK_STREAM, 0);
  11.     // 设置端口复用
  12.     int opt = 1;
  13.     setsockopt(sfd, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(int));
  14.     struct sockaddr_in soaddr;
  15.     bzero(&soaddr, sizeof(soaddr));
  16.     soaddr.sin_family = AF_INET;
  17.     soaddr.sin_port = htons(9999);
  18.     soaddr.sin_addr.s_addr = htonl(INADDR_ANY);
  19.     Bind(sfd, (struct sockaddr *)&soaddr, sizeof(soaddr));
  20.     // 监听-listen
  21.     Listen(sfd, 128);
  22.     struct pollfd connfd[POLLLEN];
  23.     int maxfd;  // 当前需要监听的最大的文件描述符
  24.     int nready; // 返回的需要处理的个数
  25.     int cfd;    // 通信描述符
  26.     int i, maxi = 0;
  27.     struct sockaddr_in clientsocket;
  28.     socklen_t clilen;
  29.     char buff[64]; // 通信数据
  30.     // 初始化有效的文件描述符数组
  31.     for (int i = 0; i < POLLLEN; i++)
  32.     {
  33.         bzero(&connfd[i], sizeof(connfd[i]));
  34.         connfd[i].fd = -1;
  35.     }
  36.     maxfd = sfd;
  37.     connfd[0].fd = sfd;
  38.     connfd[0].events = POLLIN;
  39.     while (1)
  40.     {
  41.         clilen = sizeof(clientsocket);
  42.         bzero(&clientsocket, clilen);
  43.         nready = poll(connfd, maxfd + 1, -1);
  44.         if (nready < 0)
  45.         {
  46.             if (errno == EINTR) // 被信号中断
  47.             {
  48.                 continue;
  49.             }
  50.             perror("poll error");
  51.             break;
  52.         }
  53.         if (nready > 0)
  54.         {
  55.             if (connfd[0].revents == POLLIN)
  56.             {
  57.                 // 有连接到来
  58.                 cfd = Accept(sfd, (struct sockaddr *)&clientsocket, &clilen);
  59.                 for (i = 0; i < POLLLEN; i++)
  60.                 {
  61.                     if (connfd[i].fd == -1)
  62.                     {
  63.                         connfd[i].events = POLLIN;
  64.                         connfd[i].fd = cfd;
  65.                         maxi = i;
  66.                         if (maxfd < cfd)
  67.                         {
  68.                             maxfd = cfd;
  69.                         }
  70.                         // 打印客户端的IP和PORT
  71.                         char sIP[16];
  72.                         memset(sIP, 0x00, sizeof(sIP));
  73.                         printf("client [%s:%d] connect\n", inet_ntop(AF_INET, &clientsocket.sin_addr.s_addr, sIP, sizeof(sIP)), htons(clientsocket.sin_port));
  74.                         break;
  75.                     }
  76.                 }
  77.                 if (i == POLLLEN)
  78.                 {
  79.                     close(cfd);
  80.                     printf("connect too much, server busy\n");
  81.                     continue;
  82.                 }
  83.                 if (--nready == 0)
  84.                 {
  85.                     continue;
  86.                 }
  87.             }
  88.             for (i = 1; i <= maxi; i++)
  89.             {
  90.                 if (connfd[i].fd == -1)
  91.                 {
  92.                     continue;
  93.                 }
  94.                 if (connfd[i].revents == POLLIN)
  95.                 {
  96.                     // 有数据发送过来
  97.                     int n;
  98.                     int sockfd = connfd[i].fd;
  99.                     memset(buff, 0x00, sizeof(buff));
  100.                     n = Read(sockfd, buff, sizeof(buff));
  101.                     if (n < 0)
  102.                     {
  103.                         perror("read over");
  104.                         close(sockfd);
  105.                         bzero(&connfd[i], sizeof(bzero));
  106.                         connfd[i].fd = -1; // 将connfd[i]置为-1,表示该位置可用
  107.                     }
  108.                     else if (n == 0)
  109.                     {
  110.                         // printf("client is closed\n");
  111.                         close(sockfd);
  112.                         bzero(&connfd[i], sizeof(bzero));
  113.                         connfd[i].fd = -1; // 将connfd[i]置为-1,表示该位置可用
  114.                     }
  115.                     else
  116.                     {
  117.                         printf("[%d]:[%s]\n", n, buff);
  118.                         for (i = 0; i < n; i++)
  119.                         {
  120.                             buff[i] = toupper(buff[i]);
  121.                         }
  122.                         Write(sockfd, buff, n);
  123.                     }
  124.                     if (--nready <= 0)
  125.                     {
  126.                         break; // 注意这里是break,而不是continue, 应该是从最外层的while继续循环
  127.                     }
  128.                 }
  129.             }
  130.         }
  131.     }
  132.     close(sfd);
  133.     return 0;
  134. }
复制代码
保举一个零声教育学习教程,个人以为老师讲得不错,分享给大家:[Linux,Nginx,ZeroMQ,MySQL,Redis,
fastdfs,MongoDB,ZK,流媒体,CDN,P2P,K8S,Docker,
TCP/IP,协程,DPDK等技术内容,点击立即学习:链接

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

立山

金牌会员
这个人很懒什么都没写!

标签云

快速回复 返回顶部 返回列表