gpt4 book ai didi

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

转载 作者:太空宇宙 更新时间:2023-11-04 06:25:11 24 4
gpt4 key购买 nike

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

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/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com