我正在使用选择系统等待输入。我也在循环中执行此操作。 这是代码。
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/