c - select() 系统调用未按预期工作

标签 c unix network-programming system-calls

我正在使用选择系统等待输入。我也在循环中执行此操作。 这是代码。

int main()
{
    fd_set rfds;
    struct timeval tv;  

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

    tv.tv_sec = 5;
        tv.tv_usec = 0;

    while(1)
    {
        select(1,&rfds,NULL,NULL,&tv);
        if(FD_ISSET(0,&rfds))
        {
            write(STDOUT_FILENO,"yes",3);
            FD_CLR(0,&rfds);    
        }

        tv.tv_sec = 5;
            tv.tv_usec = 0; 
    }   
    return 0;

}

现在的问题是 select 调用仅在第一次时工作正常。如果我在前 5 秒内提供输入,我得到 yes 作为输出,但在接下来的迭代中 fd(0) 保持未设置,无论我是否是否提供任何输入。知道如何解决这个问题。

最佳答案

您的代码有两个问题。

问题一:选择空文件描述符集

首先是 select 修改了给定的文件描述符集——在 select 返回后,它们包含准备好进行 I/O 的文件描述符。这意味着如果超时没有对 stdin 进行任何输入,rfds 将为空,并且下一次调用 select 将等待空文件描述符集上的输入——它永远找不到的地方。

stdin 上的输入将在集合中保留 STDIN_FILENO(即 0),但是如果它出现,您将调用

FD_CLR(0,&rfds);

要从集合中删除标准输入,select 也会在这种情况下等待一个空的 fd 集合。不过,我知道你为什么把它放在那里,它与第二个问题有关(见下文)。无论如何,第一个问题的解决方法是在再次调用 select 之前将 stdin 放回 fd 集中:

FD_SET(0, &rfds);

问题二:输入永远保存在标准输入中

第二个问题是,当您的程序等待 stdin 上的输入时,它从不使用任何输入。这意味着如果您在每次使用 FD_SET(0, &rfds); 调用 select 之前修复文件描述符集,您的程序最终将一遍又一遍地打印“yes”在无限循环中。

这是因为一旦输入等待在 stdin 中被使用,在该文件描述符上调用 select 将导致 select 检查是否有输入在等待,识别是的,有,告诉你这个事实,于是你不对它做任何事情,只是要求 select 检查它是否仍然存在。无论您多久检查一次,它都是什么。

我不知道你到底想如何使用输入,所以接下来的部分是猜测。我假设您希望程序在用户输入内容时写"is"。当用户输入某些内容时,并不总是明确定义,但一种常见的解释是用户输入通常是基于行的——用户希望在他按下回车键后事情就会发生。那么,一种明智的方法是在数据出现时丢弃直到下一个换行符,以便每次按下返回按钮都会产生"is"。这可能看起来像这样:

while ((c = getchar()) != EOF) && c != '\n'); // discard until end-of-line

使用 int c;

整合

总而言之,这可能会做你想做的事:

#include <sys/select.h>
#include <unistd.h>
#include <stdio.h>

int main()
{
  fd_set rfds;
  struct timeval tv;
  int c;

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

  tv.tv_sec = 5;
  tv.tv_usec = 0;

  while(1)
  {
    select(1,&rfds,NULL,NULL,&tv);
    if(FD_ISSET(0, &rfds))
    {
      puts("yes");
      while((c = getchar()) != EOF && c != '\n'); // consume input
    } else {
      puts("no");
      FD_SET(0, &rfds); // place stdin back in the fd set
    }

    tv.tv_sec = 5;
    tv.tv_usec = 0; 
  }

  return 0;
}

关于c - select() 系统调用未按预期工作,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/27988321/

相关文章:

c++ - OpenMP 和 VS 2012 发现意外的文件结尾

c++ - 为什么在宏中使用明显无意义的 do-while 和 if-else 语句?

c - 如何从 .t​​ext 文件中查找最小值和最大值

bash - 使用 tr 删除特定字符串

python - 读取 .tar 文件中的 .gz 文件而不解压

parsing - 在 TLS 协议(protocol)中解析扩展问候消息的正确方法是什么?

c - 逗号表达式的左侧操作数无效

linux - 如何使用 MobaXTerm 便携版查找空目录

bash - Bash-循环通过ARP表并将IP存储在变量中

c# - 计算 TcpClient 通过 NetworkStream BinaryReader/BinaryWriter 传输的字节数