题
我在尝试弄清楚如何使用 scanf()
时遇到了很多问题。它似乎与整数一样正常,相当直接 scanf("%d",& i)
。
我遇到的问题是在尝试读取输入的循环中使用 scanf()
。例如:
do {
printf("counter: %d: ", counter);
scanf("%c %c%d", &command, &prefix, &input);
} while (command != 'q');
-
当我输入有效结构的输入,如
c P101
时,它似乎再次循环,然后再提示我。即使只有一个,这似乎也会发生:scanf("%c", &c)
在while循环中。在再次提示我之前,它会进行两次循环。是什么让它循环两次,我该如何阻止它呢?
-
当我输入较少量的输入时,编程上没有其他字符或数字,如
q
,按Enter似乎提示我输入更多。如何让scanf()
处理单字符和双字符条目?
醇>
解决方案
当您输入“ c P101
”时该程序实际上收到“ c P101 \ n
”。大多数转换说明符都跳过前导空格,包括换行符,但%c
却没有。第一次围绕所有内容直到“ \ n
”被读取,第二次围绕“\ n”,被读入 command
," c
"读入前缀
和“ P
”留下的不是数字,因此转换失败并且“ P101 \ n
”留在溪边。下一次“ P
”存储在命令中,“ 1
”存储在前缀中,并且 1
(来自剩余的“ 01
”)存储到具有“ \ n
”的输入中。仍然在下一次的流。您可以通过在格式字符串的开头添加一个空格来解决此问题,该字符串将跳过任何前导空格,包括换行符。
当你输入“ q
”,“ q \ n
”时,第二种情况就会发生类似的情况。输入到流中,第一次围绕“ q
”输入。被读取,第二次是“ \ n
”被读取,仅在第三次调用时是第二个“ q
”阅读,您可以通过在格式字符串的开头添加空格字符来再次避免此问题。
更好的方法是使用类似fgets()的方法一次处理一行,然后使用sscanf()进行解析。
其他提示
它真的坏了!我不知道
#include <stdio.h>
int main(void)
{
int counter = 1;
char command, prefix;
int input;
do
{
printf("counter: %d: ", counter);
scanf("%c %c%d", &command, &prefix, &input);
printf("---%c %c%d---\n", command, prefix, input);
counter++;
} while (command != 'q');
}
counter: 1: a b1
---a b1---
counter: 2: c d2
---
c1---
counter: 3: e f3
---d 21---
counter: 4: ---e f3---
counter: 5: g h4
---
g3---
输出似乎符合罗伯特的答案。
一旦你有包含该行的字符串。即“C P101”,您可以使用sscanf的解析功能。
请参阅: http://www.cplusplus.com/reference/clibrary/cstdio/sscanf。 HTML
对于问题1,我怀疑你的 printf()
有问题,因为没有终止“\ n”。
printf
的默认行为是缓冲输出,直到它有一个完整的行。除非您明确更改 stdout
上的缓冲。
对于问题2,您只是遇到了 scanf()
的一个最大问题。除非您的输入与您指定的扫描字符串完全匹配,否则您的结果将与您的预期完全不同。
如果你有一个选项,你可以忽略 scanf()
并进行自己的解析,从而获得更好的结果(以及更少的安全问题)。例如,使用 fgets()
将整行读入字符串,然后处理字符串的各个字段&#8212;甚至可能使用 sscanf()
。
也许使用while循环,而不是do ... while循环会有所帮助。这样,在执行代码之前测试条件。 请尝试以下代码段:
while(command != 'q')
{
//statements
}
另外,如果您提前知道字符串长度,'for'循环可以比'while'循环更容易使用。还有一些更复杂的方法来动态确定长度。
作为最后的咆哮: scanf()
不会“吮吸”。它做它做的事,就是这样。 fgets()
非常危险(虽然对于无风险的应用程序很方便),因为它本身不会对输入进行任何检查。它通常被称为利用点,特别是缓冲区溢出攻击,覆盖未分配给该变量的寄存器中的空间。因此,如果您选择使用它,请花一些时间进行一些可靠的错误检查/纠正。
希望这可以帮助将来的人,因为我认为你现在已经处理好了! ;)