- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
根据 Linux Programmer's Manual , poll
可以等待一组文件描述符中的一个准备好执行 I/O。
据我了解,如果我将POLLIN
添加到events
,poll
将返回> 0
整数,当至少有一个 fd 准备好被读取时。
考虑以下代码,在这段代码中,我希望程序在我键入字符 \n
后立即回显我的输入。
int main(){
char buffer[maxn];
while (true) {
struct pollfd pfd[1];
std::memset(pfd, 0, sizeof pfd);
pfd[0].fd = STDIN_FILENO;
pfd[0].events = POLLIN;
int ret = poll(pfd, 1, 1000);
if (ret < 0) {
}
else if (ret == 0) {
}
else {
if ((pfd[0].revents & POLLIN) == POLLIN) {
int n;
n = fscanf(stdin, "%s", &buffer);
if(n > 0){
printf("data from stdin: %s\n", buffer);
}
}else if((pfd[1].revents & POLLHUP) == POLLHUP){
break;
}
}
}
}
当我打字时
aa bb cc dd
我以为 fscanf
没有从标准输入中检索所有数据,因为它只读取 aa
。所以当循环重新开始时,stdin 的 fd 应该仍然准备就绪。因此,(pfd[0].revents & POLLIN) == POLLIN
仍然有效,所以我认为我们可以看到以下输出
data from stdin: aa
data from stdin: bb
data from stdin: cc
data from stdin: dd
然而,实际上只打印了第一行。我在这里很奇怪,我认为这与 epoll
的 Edge-triggered mode 类似.但是,poll
是级别触发的。
那么您能解释一下为什么 fscanf
会发生这种情况吗?
最佳答案
轮询在文件描述符 级别进行,而fscanf
在更高的文件句柄级别工作。
在较高级别,C 运行时库可以自由缓存输入流,这种方式会影响您在较低级别看到的内容。
例如(这可能就是这里发生的事情),你第一次fscanf
你的话aa
,整个行从文件描述符中读取并缓存,然后第一个单词返回给您。
后续fscanf
(没有干预 poll
)将首先检查缓存以获取下一个单词,如果不存在,它将返回到文件描述符以获取更多输入。
不幸的是,您在执行此操作之前检查轮询事件的事实会导致问题。就文件描述符 级别而言,整个 行已被您的第一个 fscanf
读取。所以没有进一步的输入可用 - poll
因此将等到此类信息确实可用。
如果您更改:
n = fscanf(stdin, "%s", buffer);
进入:
n = read(STDIN_FILENO, buffer, 3);
并更改 printf
到:
printf("data from stdin: %*.*s\n", n, n, buffer);
在那种情况下,您确实在按下 ENTER 键后立即得到了您期望的输出:
data from stdin: aa
data from stdin: bb
data from stdin: cc
data from stdin: dd
请记住,示例代码最多读取 三个 个字符(如 aa<space>
)而不是一个单词。它更多地是为了说明问题是什么,而不是为您提供解决方案(以匹配您的问题“您能解释一下为什么会发生这种情况吗?”)。
当后者的缓存会影响前者时,解决方案是不要混合描述符和基于句柄的 I/O。
关于c++ - 为什么在这种情况下 stdin fd 还没有准备好,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49476088/
我有以下代码可以完全按预期工作: from subprocess import Popen process = Popen( ["/bin/bash"], stdin=sys.stdi
我有一个关于 php-cli 的新问题。 我正在使用这个: define("STDIN", fopen('php://stdin','r')); $input = ""; while($input =
这个问题在这里已经有了答案: Can fseek(stdin,1,SEEK_SET) or rewind(stdin) be used to flush the input buffer inste
我正在编写一个 python 程序,它将所有输入都大写(替代非工作 tr '[:lowers:]' '[:upper:]')。语言环境是 ru_RU.UTF-8,我使用 PYTHONIOENCODIN
自从我发现 fflush(stdin) 不是处理熟悉的“换行潜伏在输入缓冲区中”问题的可移植方法,我一直在使用当我必须使用scanf时如下: while((c = getchar()) != '\n'
当我使用时在 Perl 模块( *.pm )文件中,它不会从键盘读取输入,但是当我使用 时在同一个地方它工作得很好。 为什么我使用时没有得到输入? 最佳答案 STDIN 是记录的文件句柄。还有 st
stdin 是否是一个指针,正如我在 fgets() 中看到的那样。 我使用“0”作为标准输入的读取或写入错误,并在 fgets 期间出现段错误。 STDIN宏和0是否相同。 stdin 是文件指针吗
我想知道 STDIN 和 $stdin 之间是否有任何真正的区别。我在 irb: STDIN == $stdin 并返回 true。它们只是同一事物的两个名称吗?还是有什么不同? 最佳答案 来自 Ru
有没有一种简单的方法可以将内容通过管道传输到编辑器原子? 例如: echo "Content." | atom 不幸的是atom没有获取到内容。当前版本的 gedit 具有参数 - 以启用读取 STD
这个问题已经有答案了: Using fflush(stdin) (7 个回答) 已关闭 9 年前。 我有一个这样的测试代码 #include #include #include int main
我有一个 bash启动 scp 的脚本通过以下方式: echo "${SCP_PASS:-$PASSWORD}" | ( exec 3<&0; scp -qp ${SCP_PORT:+-P$SCP_P
我正在创建一个 NASM 汇编代码来从标准输入读取文件中存在的二维数字数组我正在运行这样的可执行文件 -> ./abc < input.txt . 之后,我将在终端上显示读取的二维数组,然后我想获取箭
这是一个循环,它重复地从 stdin 获取两个字符并输出它们。 char buf[2]; while (1) { printf("give me two characters: ");
我有一个 golang 程序,可以为 jq 做一个简单的 repl。 .我希望能够在程序启动时从 stdin 读取输入到一个临时文件中,这样我就可以将 repl 与管道输入一起使用。 cat file
有没有非阻塞的 PHP 从 STDIN 读取: 我试过了: stream_set_blocking(STDIN, false); echo fread(STDIN, 1); 还有这个: $stdin
这实际上与我已经回答的另一个问题有关。这个问题在这里:Redirecting stdout of one process object to stdin of another 我的问题是(我认为)获取
我只是一个java初学者,目前正在大学学习,但由于某些原因我不会深入,我无法询问我的导师。 我在 Netbeans 中使用 StdIn 库时遇到问题。在类里面我们使用 DrJava,但由于我无法让它在
Ruby 有两种引用标准输入的方法:STDIN 常量和$stdin 全局变量。 除了我可以将不同的 IO 对象分配给 $stdin 因为它不是常量(例如,在我的 child 中 fork 重定向 IO
我是 Pythonizer 的作者我正在尝试将 CGI.pm 的代码从标准 perl 库转换为 Python。我在 read_from_client 中看到这段代码: read(\*STDIN, $$
我正在使用 laravel 5.6 并遇到问题,当我在控制台中使用命令“php artisan vendor:publish”时,出现以下错误: [ERROR] Use of undefined co
我是一名优秀的程序员,十分优秀!