6

我是套接字编程的新手,我正在尝试彻底了解它是如何工作的,但现在我真的坚持使用select().

问题是在我的代码中,在 select 检测到活动并且 fd 保持设置后,似乎在下一次迭代FD_ISSET时会自动返回 true,就像它会忽略 select 函数一样。问题似乎与这个问题相同,但我做了所有我在那里找到的但无济于事:http ://compgroups.net/comp.unix.programmer/how-does-fd_isset-return-0-after-returne /55058

我确保select()在使用 Linux 之后重新初始化 timeval 变量,并且我了解此函数在不同操作系统上的行为不同,我还使用selectFD_ZERO和之前重新初始化了 fd 集。FD_SET

我究竟做错了什么?这是代码:

#include <stdio.h>
#include <strings.h>

#include <sys/select.h>
#include <sys/time.h>

int main () {
  struct timeval tv;

  tv.tv_sec = 5; // 5 seconds timeout
  tv.tv_usec = 0;

  fd_set afds, rfds;

  FD_ZERO(&afds);
  FD_SET(0, &afds);

  while (1) {
    rfds = afds;
    select(1, &rfds, NULL, NULL, &tv);
    // linux, reinitialize tv?
    tv.tv_sec = 5;
    tv.tv_usec = 0;

    // so at this point after select runs the first time and detects STDIN activity
    // it will enter an infinite loop printing "fd 0 is set" (why?)
    if (FD_ISSET(0, &rfds)) {
      printf("fd 0 is set\n");
      FD_CLR(0, &rfds);
    } else {
      printf("fd 0 is NOT set\n");
    }
  }
}

问题编辑,因为我是新用户,无法回答:

事实是我在选择之前初始化 rfds,当它被分配 afds 的值时,它又总是用 FD_ZERO(&afds); 设置。FD_SET(0, &afds); 这对我仍然不起作用。

以下是我的理解:

  1. 我将标准输入文件描述符添加到 afds

  2. 进入while无限循环,rfds = afds(循环开始时rfds永远是= afds)

  3. 此外,此时 FD_ISSET(0, &rfds) 将始终为 != 0

  4. select 有 5 秒的超时,所以此时如果我在 5 秒过去之前没有输入任何内容,它就会退出,UNSETTING FD_ISSET(0, &rfds) - 对吗?因此,如果没有输入任何内容,select 实际上会取消设置 fd 0。这似乎工作正常

  5. 当我在超时之前输入一些东西时,问题就来了。此时,FD_ISSET(0, &rfds) 返回 != 0,它打印 fd 0 is set,然后每个循环 fd 都会被设置

好的,这是准确的吗,我猜对了吗?所以实际上 select 不会等待时间过去,因为它实际上检测到 fd 已准备好并退出,设置 fd != 0 ?

这需要进一步的问题:如果我需要服务器每隔一段时间自动向多个客户端发送消息(独立于它从客户端读取的内容),是否可以通过修改代码来使用 select 和 gettimeofday以上?

谢谢您的帮助。

4

2 回答 2

16

select()是电平触发的,不是边沿触发的。当您向它传递一组文件描述符时,它会回来告诉您当前哪些是可读/可写/异常的,而不仅仅是最近更改状态的那些。

在这种情况下,每次调用时 fhe FD 都被标记为可读,select()因为当它出现时,您没有做任何事情使其不可读(例如耗尽可用输入)。

于 2013-01-23T17:20:22.613 回答
6

在 select() 触发后,存储在 fd_set 中的值会保留。在应用程序中,select() 可以监视许多套接字。自动清理 fd_set 意味着您永远无法检测到多个套接字需要服务。

您需要在无限循环中执行 FD_ZERO() 和 FD_SET() ,以便在每次通过时,在调用 select() 之前干净地初始化 fd_set。

于 2013-01-23T17:25:07.647 回答