[关闭]
@LIUHUAN 2016-05-01T16:18:29.000000Z 字数 1668 阅读 993

Linux C/C++

select 调用

1. 函数调用原型

  1. int select(int maxfdp1,fd_set *readset,fd_set *writeset,
  2. fd_set *exceptset,const struct timeval *timeout)

2.参数说明

2.1 timeout参数

结构体定义:

  1. struct timeval {
  2. long tv_sec;//秒
  3. long tv_usec;//微秒
  4. };
  • 如果这个参数不为null,且为有效的一段时间(不为0)那么select会在等待这个时间内,返回,不管有没有这三种事件发生。

  • 如果为null,那么select调用会阻塞下去,直到readset,writeset,exceptset中的对应的文件描述符上有对应的事件发生。

  • 如果这个值不为null但是里面的值为0(tv_sec=tv_usec=0),那么这个调用,会检查有没有文件描述符可以使用(对应的可读,可写,异常),然后立即返回。
    这个也叫做polling。


2.2 readset,writeset,exceptset

这三个参数分别是调用者,感兴趣文件描述符的集合,select会监控这三个集合中的文件描述符的状态,然后根据发生的事件返回,返回会调用者需要检测是read,write,except哪一种事件发生了。
下面提供了对于文件描述符集合的几个函数(实际为宏定义)

  • 从集合当中删除描述符fd
  1. void FD_CLR(int fd,fd_set *set)
  • 将fd描述符添加到集合当中
  1. void FD_SET(int fd,fd_set * set)
  • 将集合中设置监控的文件描述符清空
  1. void FD_ZERO(fd_set * set)
  • 判断fd是不是在集合当中
  1. int FD_ISSET(int fd,fd_set* set)

2.3 maxfdp1

监听文件描述符数值的最大值+1

2.4 返回值

  • 有监控的事件发生返回发生的事件的个数
  • 超时返回或者出错-1

3.使用方法

下面是一个类似可以响应多个客户端连接的服务器程序。

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <unistd.h>
  4. int main(int argc,char* argv[]){
  5. fd_set rfds;
  6. struct timeval tv;
  7. int retval;
  8. int listenedfd;
  9. FD_ZERO(&rfds);/*初始化集合*/
  10. /*添加监控的文件描述符到集合当中*/
  11. FD_SET(listenefd, &rfds);
  12. /*Wait up to five seconds.
  13. 设置超时时间
  14. */
  15. tv.tv_sec = 5;
  16. tv.tv_usec = 0;
  17. /*调用select 在超时时间tv内返回*/
  18. retval = select(listenefd + 1, &rfds, NULL, NULL, &tv);
  19. /* Don't rely on the value of tv now! */
  20. if (retval == -1)/*超时或者出错*/
  21. perror("select()");
  22. else if (retval){ /*有返回,监控的文件描述符会有相应的事件*/
  23. printf("Data is available now.\n");
  24. /**接受连接,并保存在RIOfd(感兴趣的fd)数组中,以便后来处理*/
  25. if(FD_ISSET(listenfd, &rfds)){
  26. /*接受连接*/
  27. accept_connection();
  28. /*保存连接*/
  29. save_connected_fd();
  30. }
  31. }
  32. /* FD_ISSET(0, &rfds) will be true. */
  33. /*遍历所有的感兴趣的文件描述符,是否发生了事件*/
  34. for(int i=0;i<maxRIOfdnumber;++i){
  35. if(FD_ISSET(RIOfd[i], &rfds))
  36. /*对发生的事件进行处理*/
  37. do_with(RIOfd[i]);
  38. }else
  39. printf("No data within five seconds.\n");
  40. }
  41. exit(EXIT_SUCCESS);
  42. }

4.参考内容

  • Linux man select
  • Unix 网络编程 卷I 套接字联网API
添加新批注
在作者公开此批注前,只有你和作者可见。
回复批注