UNIX网络编程——select函数的并发限制和 poll 函数应用举例

一、用select实现的并发服务器,能达到的并发数,受两方面限制


       1、一个进程能打开的最大文件描述符限制。这可以通过调整内核参数。可以通过ulimit -n来调整或者使用setrlimit函数设置, 但一个系统所能打开的最大数也是有限的,跟内存大小有关,可以通过cat /proc/sys/fs/file-max 查看

       2、select中的fd_set集合容量的限制(FD_SETSIZE,一般为1024) ,这需要重新编译内核。


可以写个测试程序,只建立连接,看看最多能够建立多少个连接,客户端程序如下:

  1. #include <sys/types.h>  
  2. #include <sys/socket.h>  
  3. #include <netinet/in.h>  
  4. #include <arpa/inet.h>  
  5. #include <signal.h>  
  6. #include <stdlib.h>  
  7. #include <stdio.h>  
  8. #include <errno.h>  
  9. #include <string.h>  
  10.   
  11. #define ERR_EXIT(m) \  
  12.         do \  
  13.         { \  
  14.                 perror(m); \  
  15.                 exit(EXIT_FAILURE); \  
  16.         } while(0)  
  17.   
  18.   
  19. int main(void)  
  20. {  
  21.     int count = 0;  
  22.     while(1)  
  23.     {  
  24.         int sock;  
  25.         if ((sock = socket(PF_INET, SOCK_STREAM, IPPROTO_TCP)) < 0)  
  26.         {  
  27.             sleep(4);  
  28.             ERR_EXIT("socket");  
  29.         }  
  30.   
  31.         struct sockaddr_in servaddr;  
  32.         memset(&servaddr, 0, sizeof(servaddr));  
  33.         servaddr.sin_family = AF_INET;  
  34.         servaddr.sin_port = htons(5188);  
  35.         servaddr.sin_addr.s_addr = inet_addr("127.0.0.1");  
  36.   
  37.         if (connect(sock, (struct sockaddr *)&servaddr, sizeof(servaddr)) < 0)  
  38.             ERR_EXIT("connect");  
  39.   
  40.         struct sockaddr_in localaddr;  
  41.         socklen_t addrlen = sizeof(localaddr);  
  42.         if (getsockname(sock, (struct sockaddr *)&localaddr, &addrlen) < 0)  
  43.             ERR_EXIT("getsockname");  
  44.   
  45.         printf("ip=%s port=%d\n", inet_ntoa(localaddr.sin_addr), ntohs(localaddr.sin_port));  
  46.         printf("count = %d\n", ++count);  
  47.   
  48.     }  
  49.   
  50.     return 0;  
  51. }  

服务器的代码serv.c:来自<<UNIX网络编程——使用select函数编写客户端和服务器>>最后的服务器程序。

  1. #include<stdio.h>  
  2. #include<sys/types.h>  
  3. #include<sys/socket.h>  
  4. #include<unistd.h>  
  5. #include<stdlib.h>  
  6. #include<errno.h>  
  7. #include<arpa/inet.h>  
  8. #include<netinet/in.h>  
  9. #include<string.h>  
  10. #include<signal.h>  
  11. #include<sys/wait.h>  
  12.   
  13.   
  14.   
  15. #define ERR_EXIT(m) \  
  16.     do { \  
  17.         perror(m); \  
  18.         exit(EXIT_FAILURE); \  
  19.     } while (0)  
  20.   
  21.   
  22. int main(void)  
  23. {  
  24.       
  25.     signal(SIGPIPE, SIG_IGN);  
  26.     int listenfd; //被动套接字(文件描述符),即只可以accept, 监听套接字  
  27.     if ((listenfd = socket(PF_INET, SOCK_STREAM, IPPROTO_TCP)) < 0)  
  28. //  listenfd = socket(AF_INET, SOCK_STREAM, 0)    
  29.         ERR_EXIT("socket error");  
  30.   
  31.     struct sockaddr_in servaddr;  
  32.     memset(&servaddr, 0, sizeof(servaddr));  
  33.     servaddr.sin_family = AF_INET;  
  34.     servaddr.sin_port = htons(5188);  
  35.     servaddr.sin_addr.s_addr = htonl(INADDR_ANY);   
  36.     /* servaddr.sin_addr.s_addr = inet_addr("127.0.0.1"); */  
  37.     /* inet_aton("127.0.0.1", &servaddr.sin_addr); */  
  38.       
  39.     int on = 1;  
  40.     if (setsockopt(listenfd, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on)) < 0)  
  41.         ERR_EXIT("setsockopt error");  
  42.   
  43.     if (bind(listenfd, (struct sockaddr*)&servaddr,sizeof(servaddr)) < 0)  
  44.         ERR_EXIT("bind error");  
  45.   
  46.     if (listen(listenfd, SOMAXCONN) < 0) //listen应在socket和bind之后,而在accept之前  
  47.         ERR_EXIT("listen error");  
  48.       
  49.     struct sockaddr_in peeraddr; //传出参数  
  50.     socklen_t peerlen = sizeof(peeraddr); //传入传出参数,必须有初始值  
  51.       
  52.     int conn; // 已连接套接字(变为主动套接字,即可以主动connect)  
  53.     int i;  
  54.     int client[FD_SETSIZE];  
  55.     int maxi = 0; // client数组中最大不空闲位置的下标  
  56.     for (i = 0; i < FD_SETSIZE; i++)  
  57.         client[i] = -1;  
  58.   
  59.     int nready;  
  60.     int maxfd = listenfd;  
  61.     fd_set rset;  
  62.     fd_set allset;  
  63.     FD_ZERO(&rset);  
  64.     FD_ZERO(&allset);  
  65.     FD_SET(listenfd, &allset);  
  66.   
  67.     int count = 0;  
  68.     while (1) {  
  69.         rset = allset;  
  70.         nready = select(maxfd + 1, &rset, NULL, NULL, NULL);  
  71.         if (nready == -1) {  
  72.             if (errno == EINTR)  
  73.                 continue;  
  74.             ERR_EXIT("select error");  
  75.         }  
  76.   
  77.         if (nready == 0)  
  78.             continue;  
  79.   
  80.         if (FD_ISSET(listenfd, &rset)) {  
  81.           
  82.             conn = accept(listenfd, (struct sockaddr*)&peeraddr, &peerlen);  //accept不再阻塞  
  83.             if (conn == -1)  
  84.                 ERR_EXIT("accept error");  
  85.             printf("count = %d\n", ++count);  
  86.             for (i = 0; i < FD_SETSIZE; i++) {  
  87.                 if (client[i] < 0) {  
  88.                     client[i] = conn;  
  89.                     if (i > maxi)  
  90.                         maxi = i;  
  91.                     break;  
  92.                 }   
  93.             }  
  94.               
  95.             if (i == FD_SETSIZE) {  
  96.                 fprintf(stderr, "too many clients\n");  
  97.                 exit(EXIT_FAILURE);  
  98.             }  
  99.   
  100.             printf("recv connect ip=%s port=%d\n", inet_ntoa(peeraddr.sin_addr),  
  101.                 ntohs(peeraddr.sin_port));  
  102.   
  103.             FD_SET(conn, &allset);  
  104.             if (conn > maxfd)  
  105.                 maxfd = conn;  
  106.   
  107.             if (--nready <= 0)  
  108.                 continue;  
  109.         }  
  110.   
  111.         for (i = 0; i <= maxi; i++) {  
  112.             conn = client[i];  
  113.             if (conn == -1)  
  114.                 continue;  
  115.   
  116.             if (FD_ISSET(conn, &rset)) {  
  117.                   
  118.                 char recvbuf[1024] = {0};  
  119.                 int ret = read(conn, recvbuf, 1024);  
  120.                 if (ret == -1)  
  121.                     ERR_EXIT("read error");  
  122.                 else if (ret  == 0) { //客户端关闭   
  123.                     printf("client close \n");  
  124.                     FD_CLR(conn, &allset);  
  125.                     client[i] = -1;  
  126.                     close(conn);  
  127.                 }  
  128.           
  129.                 fputs(recvbuf, stdout);  
  130.                 write(conn, recvbuf, strlen(recvbuf));  
  131.                   
  132.                 if (--nready <= 0)  
  133.                     break;   
  134.             }  
  135.         }  
  136.   
  137.   
  138.     }  
  139.           
  140.     return 0;  
  141. }  
  142.   
  143. /* select所能承受的最大并发数受 
  144.  * 1.一个进程所能打开的最大文件描述符数,可以通过ulimit -n来调整 
  145.  *   但一个系统所能打开的最大数也是有限的,跟内存有关,可以通过cat /proc/sys/fs/file-max 查看 
  146.  * 2.FD_SETSIZE(fd_set)的限制,这个需要重新编译内核                                                                           
  147.  */  

       先启动select 的服务器端程序,再启动客户端测试程序:

  1. huangcheng@ubuntu:~$ ./serv  
  2. count = 1  
  3. recv connect ip=127.0.0.1 port=48370  
  4. count = 2  
  5. recv connect ip=127.0.0.1 port=48371  
  6. count = 3  
  7. recv connect ip=127.0.0.1 port=48372  
  8. count = 4  
  9. recv connect ip=127.0.0.1 port=48373  
  10. ....................................  
  11. recv connect ip=127.0.0.1 port=49389  
  12. count = 1020  
  13. recv connect ip=127.0.0.1 port=49390  
  14. accept error: Too many open files  
  1. huangcheng@ubuntu:~$ ./cli  
  2. ip=127.0.0.1 port=46327  
  3. count = 1  
  4. ip=127.0.0.1 port=46328  
  5. count = 2  
  6. ip=127.0.0.1 port=46329  
  7. count = 3  
  8. ip=127.0.0.1 port=46330  
  9. count = 4  
  10. ip=127.0.0.1 port=46331  
  11. count = 5  
  12. ip=127.0.0.1 port=46332  
  13. count = 6  
  14. ip=127.0.0.1 port=46333  
  15. .......................  
  16. ip=127.0.0.1 port=47345  
  17. count = 1020  
  18. ip=127.0.0.1 port=47346  
  19. count = 1021  
  20. socket: Too many open files  

       输出太多条目,上面只截取最后几条,从中可以看出对于客户端,最多只能开启1021个连接套接字,因为总共是1024个,还得除去0、1、2。而服务器端只能accept 返回1020个已连接套接字,因为除了0、1、2之外还有一个监听套接字,客户端某一个套接字(不一定是最后一个)虽然已经建立了连接,在已完成连接队列中,但accept 返回时达到最大描述符限制,返回错误,打印提示信息。


       也许有人会注意到上面有一行 sleep(4);当客户端调用socket准备创建第1022个套接字时,如上所示也会提示错误,此时socket函数返回-1出错,如果没有睡眠4s后再退出进程会有什么问题呢?如果直接退出进程,会将客户端所打开的所有套接字关闭掉,即向服务器端发送了很多FIN段,而此时也许服务器端还一直在accept ,即还在从已连接队列中返回已连接套接字,此时服务器端除了关心监听套接字的可读事件,也开始关心前面已建立连接的套接字的可读事件,read 返回0,所以会有很多 client close 字段 参杂在条目的输出中,还有个问题就是,因为read 返回0,服务器端会将自身的已连接套接字关闭掉,那么也许刚才说的客户端某一个连接会被accept 返回,即测试不出服务器端真正的并发容量。

  1. huangcheng@ubuntu:~$ ./serv  
  2. count = 1  
  3. recv connect ip=127.0.0.1 port=50413  
  4. count = 2  
  5. ....................................  
  6. client close  
  7. client close  
  8. client close  
  9. client close  
  10. ...................................  
  11. recv connect ip=127.0.0.1 port=51433  
  12. client close  
  13. count = 1021  
  14. recv connect ip=127.0.0.1 port=51364  
  15. client close  
  16. client close  
      可以看到输出参杂着client close,且这次的count 达到了1021,原因就是服务器端前面已经有些套接字关闭了,所以accept 创建套接字不会出错,服务器进程也不会因为出错而退出,可以看到最后接收到的一个连接端口是51364,即不一定是客户端的最后一个连接。


二、poll 函数应用举例

  1. #include <poll.h>  
  2. int poll(struct pollfd *fds, nfds_t nfds, int timeout);  
参数1:结构体数组指针

  1. struct pollfd {  
  2.     int   fd;         /* file descriptor */  
  3.     short events;     /* requested events */  
  4.     short revents;    /* returned events */  
  5. };  

结构体中的fd 即套接字描述符,events 即感兴趣的事件,如下图所示,revents 即返回的事件。


参数2:结构体数组的成员个数,即文件描述符个数。

参数3:即超时时间,若为-1,表示永不超时。


       poll 跟 select 还是很相似的,比较重要的区别在于poll 所能并发的个数跟FD_SETSIZE无关,只跟一个进程所能打开的文件描述符个数有关,可以在select 程序的基础上修改成poll 程序,在运行服务器端程序之前,使用ulimit -n 2048 将限制改成2048个,注意在运行客户端进程的终端也需更改,因为客户端也会有所限制,这只是临时性的更改,因为子进程会继承这个环境参数,而我们是在bash命令行启动程序的,故在进程运行期间,文件描述符的限制为2048个。

使用poll 函数的服务器端程序如下:

  1. #include<stdio.h>  
  2. #include<sys/types.h>  
  3. #include<sys/socket.h>  
  4. #include<unistd.h>  
  5. #include<stdlib.h>  
  6. #include<errno.h>  
  7. #include<arpa/inet.h>  
  8. #include<netinet/in.h>  
  9. #include<string.h>  
  10. #include<signal.h>  
  11. #include<sys/wait.h>  
  12. #include<poll.h>  
  13.   
  14. #define ERR_EXIT(m) \  
  15.     do { \  
  16.         perror(m); \  
  17.         exit(EXIT_FAILURE); \  
  18.     } while (0)  
  19.   
  20.   
  21. int main(void)  
  22. {  
  23.     int count = 0;  
  24.     signal(SIGPIPE, SIG_IGN);  
  25.     int listenfd; //被动套接字(文件描述符),即只可以accept, 监听套接字  
  26.     if ((listenfd = socket(PF_INET, SOCK_STREAM, IPPROTO_TCP)) < 0)  
  27.         //  listenfd = socket(AF_INET, SOCK_STREAM, 0)  
  28.         ERR_EXIT("socket error");  
  29.   
  30.     struct sockaddr_in servaddr;  
  31.     memset(&servaddr, 0, sizeof(servaddr));  
  32.     servaddr.sin_family = AF_INET;  
  33.     servaddr.sin_port = htons(5188);  
  34.     servaddr.sin_addr.s_addr = htonl(INADDR_ANY);  
  35.     /* servaddr.sin_addr.s_addr = inet_addr("127.0.0.1"); */  
  36.     /* inet_aton("127.0.0.1", &servaddr.sin_addr); */  
  37.   
  38.     int on = 1;  
  39.     if (setsockopt(listenfd, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on)) < 0)  
  40.         ERR_EXIT("setsockopt error");  
  41.   
  42.     if (bind(listenfd, (struct sockaddr *)&servaddr, sizeof(servaddr)) < 0)  
  43.         ERR_EXIT("bind error");  
  44.   
  45.     if (listen(listenfd, SOMAXCONN) < 0) //listen应在socket和bind之后,而在accept之前  
  46.         ERR_EXIT("listen error");  
  47.   
  48.     struct sockaddr_in peeraddr; //传出参数  
  49.     socklen_t peerlen = sizeof(peeraddr); //传入传出参数,必须有初始值  
  50.   
  51.     int conn; // 已连接套接字(变为主动套接字,即可以主动connect)  
  52.     int i;  
  53.   
  54.     struct pollfd client[2048];  
  55.     int maxi = 0; //client[i]最大不空闲位置的下标  
  56.   
  57.     for (i = 0; i < 2048; i++)  
  58.         client[i].fd = -1;  
  59.   
  60.     int nready;  
  61.     client[0].fd = listenfd;  
  62.     client[0].events = POLLIN;  
  63.   
  64.     while (1)  
  65.     {  
  66.         /* poll检测[0, maxi + 1) */  
  67.         nready = poll(client, maxi + 1, -1);  
  68.         if (nready == -1)  
  69.         {  
  70.             if (errno == EINTR)  
  71.                 continue;  
  72.             ERR_EXIT("poll error");  
  73.         }  
  74.   
  75.         if (nready == 0)  
  76.             continue;  
  77.   
  78.         if (client[0].revents & POLLIN)  
  79.         {  
  80.   
  81.             conn = accept(listenfd, (struct sockaddr *)&peeraddr, &peerlen); //accept不再阻塞  
  82.             if (conn == -1)  
  83.                 ERR_EXIT("accept error");  
  84.   
  85.             for (i = 1; i < 2048; i++)  
  86.             {  
  87.                 if (client[i].fd < 0)  
  88.                 {  
  89.                     client[i].fd = conn;  
  90.                     if (i > maxi)  
  91.                         maxi = i;  
  92.                     break;  
  93.                 }  
  94.             }  
  95.   
  96.             if (i == 2048)  
  97.             {  
  98.                 fprintf(stderr, "too many clients\n");  
  99.                 exit(EXIT_FAILURE);  
  100.             }  
  101.   
  102.             printf("count = %d\n", ++count);  
  103.             printf("recv connect ip=%s port=%d\n", inet_ntoa(peeraddr.sin_addr),  
  104.                    ntohs(peeraddr.sin_port));  
  105.   
  106.             client[i].events = POLLIN;  
  107.   
  108.             if (--nready <= 0)  
  109.                 continue;  
  110.         }  
  111.   
  112.         for (i = 1; i <= maxi; i++)  
  113.         {  
  114.             conn = client[i].fd;  
  115.             if (conn == -1)  
  116.                 continue;  
  117.             if (client[i].revents & POLLIN)  
  118.             {  
  119.   
  120.                 char recvbuf[1024] = {0};  
  121.                 int ret = read(conn, recvbuf, 1024);  
  122.                 if (ret == -1)  
  123.                     ERR_EXIT("readline error");  
  124.                 else if (ret  == 0)   //客户端关闭  
  125.                 {  
  126.                     printf("client  close \n");  
  127.                     client[i].fd = -1;  
  128.                     close(conn);  
  129.                 }  
  130.   
  131.                 fputs(recvbuf, stdout);  
  132.                 write(conn, recvbuf, strlen(recvbuf));  
  133.   
  134.                 if (--nready <= 0)  
  135.                     break;  
  136.             }  
  137.         }  
  138.   
  139.   
  140.     }  
  141.   
  142.     return 0;  
  143. }  
  144.   
  145. /* poll 只受一个进程所能打开的最大文件描述符限制,这个可以使用ulimit -n调整 */  
参照前面对 select 函数 的解释不难理解上面的程序,就不再赘述了。来看一下输出:


  1. root@ubuntu:/home/huangcheng# ulimit -n 2048  
  2. root@ubuntu:/home/huangcheng# su - huangcheng  
  3. huangcheng@ubuntu:~$ ulimit -n  
  4. 2048  
  5. huangcheng@ubuntu:~$ ./serv  
  6. ...........................  
  7. count = 2042  
  8. recv connect ip=127.0.0.1 port=54499  
  9. count = 2043  
  10. recv connect ip=127.0.0.1 port=54500  
  11. count = 2044  
  12. recv connect ip=127.0.0.1 port=54501  
  13. accept error: Too many open files  
  1. root@ubuntu:/home/huangcheng# ulimit -n 2048  
  2. root@ubuntu:/home/huangcheng# su - huangcheng  
  3. huangcheng@ubuntu:~$ ulimit -n  
  4. 2048  
  5. huangcheng@ubuntu:~$./cli  
  6. ..........................  
  7. ip=127.0.0.1 port=54499  
  8. count = 2043  
  9. ip=127.0.0.1 port=54500  
  10. count = 2044  
  11. ip=127.0.0.1 port=54501  
  12. count = 2045  
  13. socket: Too many open files  
       可以看到现在最大的连接数已经是2045个了,虽然服务器端有某个连接没有accept 返回。即poll 比 select 能够承受更多的并发连接,只受一个进程所能打开的最大文件描述符个数限制。可以通过ulimit -n  修改,但一个系统所能打开的文件描述符个数也是有限的,这跟系统的内存大小有关系,所以说也不是可以无限地并 发,可以查看一下本机的容量:

  1. huangcheng@ubuntu:~$ cat /proc/sys/fs/file-max  
  2. 101598  
本机是虚拟机,内存2G,能够打开的文件描述符个数大约在10w个左右。
<think>好的,我现在需要给用户介绍一下C语言中的poll函数。首先,我得回忆一下自己对poll函数的理解,确保信息准确。记得poll函数是用来进行I/O多路复用的,类似于select,但可能更高效一些。不过具体的参数用法可能需要查阅资料确认。 首先,用户可能想知道poll的作用,应该先解释它的基本功能用途。比如,用于同时监视多个文件描述符的状态,看是否有可读、可写或错误等事件发生。这样可以在单线程中处理多个I/O操作,提高效率。 接下来,需要说明poll函数的原型,参数的含义。记得pollfd结构体包含文件描述符fd,关注的事件events,以及实际发生的事件revents。事件类型比如POLLIN、POLLOUT等需要解释清楚,这样用户知道如何设置检查这些标志。 然后,要对比一下pollselect的区别。比如select有文件描述符数量的限制,而poll没有,因为它是基于链表存储的。另外,select每次调用后需要重新设置fd_set,而poll通过revents字段分离了关注的事件发生的事件,可能更方便。 还要提到poll的返回值,成功时返回就绪的文件描述符数量,超时返回0,错误返回-1。参数中的超时时间设置,单位是毫秒,这点需要强调,避免用户误解。 另外,可能存在的缺点,比如在大量文件描述符时性能下降,这时候可能用epoll更好,尤其是在Linux系统下。这也是需要提及的,让用户知道不同场景下的选择。 然后考虑举例说明,比如如何初始化pollfd数组,调用poll函数,然后遍历检查每个文件描述符的事件。示例代码可能需要用中文注释,但用户可能需要具体的代码结构,比如: struct pollfd fds[2]; fds[0].fd = fd1; fds[0].events = POLLIN; fds[1].fd = fd2; fds[1].events = POLLOUT; int ret = poll(fds, 2, 1000); if (ret > 0) { // 检查每个fd的revents } 需要注意,示例代码不需要完整,但结构要清晰。同时,提醒用户处理返回值,检查错误情况,比如EINTR错误可能需要重新调用poll。 另外,可能需要提到poll在不同平台上的兼容性,比如在Windows上的支持情况,或者在某些Unix-like系统上的行为差异。不过一般来说,poll在POSIX系统中是支持的,而select可能更广泛,但poll在某些情况下更优。 最后,总结一下适用场景,比如当需要同时处理多个文件描述符,且数量不是特别大时,poll是一个不错的选择。但如果需要更高的性能,尤其是在高并发环境下,可能需要考虑epoll或kqueue。 现在需要组织这些信息,结构清晰,分点说明,确保用户能够逐步理解。同时,按照系统指令,数学公式要用$...$,但在这个问题中可能用不到数学公式,但如果有需要举例变量或表达式时要注意格式。不过这里主要是代码系统调用,所以可能不需要数学表达式,但需要检查是否有必要的地方。 确保使用中文回答,语法正确,避免术语错误。比如“文件描述符”而不是“文件描述符符”。另外,检查函数原型是否正确,参数顺序是否正确,比如poll函数的第一个参数是struct pollfd数组,第二个是nfds,第三个是timeout。 可能还需要提到头文件#include <poll.h>,这是使用poll函数必须包含的。另外,错误处理,比如当poll返回-1时,使用perror或检查errno来确定错误原因。 现在综合这些思考,组织成一个结构化的回答,分步骤介绍poll函数的作用、函数原型、参数、事件类型、与select的对比、使用示例、注意事项等。确保信息准确,帮助用户理解如何使用poll函数以及其适用场景。</think>### C语言 `poll` 函数详解 `poll` 函数是C语言中用于 **I/O多路复用** 的系统调用,允许程序同时监视多个文件描述符(如套接字、管道等),检测其是否可读、可写或发生错误。与 `select` 函数类似,但 `poll` 在处理大量文件描述符时更具灵活性效率。 --- #### 一、函数原型与头文件 ```c #include <poll.h> int poll(struct pollfd *fds, nfds_t nfds, int timeout); ``` --- #### 二、参数解析 1. **`fds`** - 类型:`struct pollfd` 数组的指针。 - 作用:指向一组需要监视的文件描述符的结构体数组。 - `struct pollfd` 定义: ```c struct pollfd { int fd; // 文件描述符 short events; // 需要监视的事件(如可读、可写) short revents; // 实际发生的事件(由内核填充) }; ``` 2. **`nfds`** - 类型:`nfds_t`(通常为无符号整型)。 - 作用:指定 `fds` 数组的长度(即监视的文件描述符数量)。 3. **`timeout`** - 类型:整型(毫秒)。 - 作用:设置超时时间: - **`> 0`**:等待指定毫秒后返回。 - **`= 0`**:立即返回(非阻塞模式)。 - **`= -1`**:阻塞等待,直到有事件发生。 --- #### 三、事件类型(`events` `revents`) - **常用事件标志**: - `POLLIN`:文件描述符可读(包括新数据到达或连接关闭)。 - `POLLOUT`:文件描述符可写。 - `POLLERR`:发生错误。 - `POLLHUP`:连接挂起(如对端关闭)。 - `POLLNVAL`:文件描述符未打开(非法)。 --- #### 四、返回值 - **`> 0`**:有事件发生的文件描述符数量。 - **`= 0`**:超时且无事件发生。 - **`-1`**:发生错误,可通过 `errno` 获取具体原因(如 `EINTR` 表示被信号中断)。 --- #### 五、`poll` vs. `select` | 特性 | `poll` | `select` | |--------------------|-----------------------------------|-----------------------------------| | 文件描述符上限 | 无限制(基于链表) | 受 `FD_SETSIZE` 限制(通常1024) | | 事件分离 | 通过 `events` `revents` 分离 | 每次调用需重新设置 `fd_set` | | 性能 | 更高效(避免重复拷贝) | 较低(频繁重置参数) | --- #### 六、使用示例 ```c #include <stdio.h> #include <poll.h> int main() { struct pollfd fds[1]; fds[0].fd = STDIN_FILENO; // 监视标准输入 fds[0].events = POLLIN; // 关注可读事件 int timeout = 5000; // 超时5秒 int ret = poll(fds, 1, timeout); if (ret == -1) { perror("poll error"); } else if (ret == 0) { printf("Timeout, no data received.\n"); } else { if (fds[0].revents & POLLIN) { printf("Standard input has data to read.\n"); } } return 0; } ``` --- #### 七、注意事项 1. **跨平台兼容性** - `poll` 在大多数Unix-like系统(如Linux、macOS)中可用,但Windows的兼容性较差(需使用 `WSAPoll`)。 2. **性能问题** - 当监视数千个文件描述符时,`poll` 的效率可能下降(需遍历所有描述符)。此时可考虑使用更高效的 `epoll`(Linux)或 `kqueue`(BSD)。 3. **错误处理** - 若返回 `-1`,需检查 `errno`: - `EINTR`:被信号中断,可安全重试。 - `EFAULT`:`fds` 指向非法内存。 --- #### 八、适用场景 - 需要同时监视少量到中等数量的文件描述符。 - 需要非阻塞I/O操作或精确控制超时时间。 - 对跨平台兼容性要求不高时。 通过合理使用 `poll`,可以显著提升程序的I/O处理效率,避免阻塞等待单个文件描述符的问题。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值