- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
当我尝试读取/解析输入时,我的C程序出现问题。
救命?
这是一个常见问题解答条目。
StackOverflow有许多与读取C语言中的输入有关的问题,答案通常集中在特定用户的特定问题上,而没有真正描绘出整个画面。
这是一种尝试全面涵盖许多常见错误的尝试,因此可以通过将这些特殊问题标记为该问题的重复项来回答这些特定的问题:
为什么最后一行打印两次?
为什么我的scanf("%d", ...)
/ scanf("%c", ...)
失败?
为什么gets()
崩溃?
...
答案被标记为社区Wiki。随时改进和(谨慎地)扩展。
最佳答案
初学者的C输入入门
文本模式与二进制模式
检查fopen()是否失败
陷阱
检查您调用的任何功能是否成功
EOF或“为什么最后一行打印两次”
永远不要使用gets()
永远不要在stdin
上使用fflush()或任何其他可供阅读的流
不要将*scanf()用于可能格式错误的输入
当*scanf()不能按预期工作时
阅读,然后解析
通过fgets()读取(部分)输入行
在内存中解析行
清理
文本模式与二进制模式
完全按写入方式读取“二进制模式”流。但是,可能会(或可能不会)在流的末尾附加实现定义的数量的空字符('\0
')。
“文本模式”流可以进行许多转换,包括(但不限于):
在行尾之前删除空格;
将换行符('\n'
)更改为输出中的其他内容(例如Windows中的"\r\n"
),然后将其更改为输入中的'\n'
;
添加,更改或删除既不是打印字符(isprint(c)
为真),水平制表符也不是换行符的字符。
很明显,文本模式和二进制模式不会混合使用。在文本模式下打开文本文件,在二进制模式下打开二进制文件。
检查fopen()是否失败
尝试打开文件可能由于各种原因而失败-缺少权限,或者找不到文件是最常见的文件。在这种情况下,fopen()将返回NULL
指针。在尝试读取或写入文件之前,请始终检查fopen
是否返回了NULL
指针。
当fopen
失败时,通常会设置全局errno变量以指示失败的原因。 (从技术上讲,这不是C语言的要求,但是POSIX和Windows都保证可以做到这一点。)errno
是可以与errno.h
中的常量进行比较的代码,但是在简单的程序中,通常您所需要的要做的就是将其转换为错误消息并使用perror()
或strerror()
打印。错误消息还应包括您传递给fopen
的文件名;如果您不这样做,当文件名不是您想的那样时,您会感到非常困惑。
#include <stdio.h>
#include <string.h>
#include <errno.h>
int main(int argc, char **argv)
{
if (argc < 2) {
fprintf(stderr, "usage: %s file\n", argv[0]);
return 1;
}
FILE *fp = fopen(argv[1], "rb");
if (!fp) {
// alternatively, just `perror(argv[1])`
fprintf(stderr, "cannot open %s: %s\n", argv[1], strerror(errno));
return 1;
}
// read from fp here
fclose(fp);
return 0;
}
true
。对“达到” EOF实际上意味着什么的误解使许多初学者写了这样的东西:
// BROKEN CODE
while (!feof(fp)) {
fgets(buffer, BUFFER_SIZE, fp);
printf("%s", buffer);
}
buffer
的内容不变,然后再次打印。
fgets
是否直接失败:
// GOOD CODE
while (fgets(buffer, BUFFER_SIZE, fp)) {
printf("%s", buffer);
}
stdin
上使用
fflush()或任何其他可供阅读的流
fflush(stdin)
放弃尚未读取的用户输入。它不会那样做。在普通ISO C中,在输入流上调用
fflush()具有
undefined behaviour。它在POSIX和MSVC中确实具有明确定义的行为,但是没有一个使它放弃尚未读取的用户输入。
int c;
do c = getchar(); while (c != EOF && c != '\n');
[
,
c
和
n
转换)。 (请参阅下一段。)
' '
,
'\n'
,...)。
"%d"
等)或字符串(
"%s"
)在任何空白处停止。虽然大多数
*scanf()
转换说明符会跳过输入中的前导空格,但
[
,
c
和
n
不会。因此,换行符仍然是第一个待处理的输入字符,从而使
%c
和
%[
都不匹配。
'\n'
)。如果不是全部都适合,那么您正在看的部分阅读的行。
d
,
i
,
u
,
o
,
x
,
a
,
e
,
f
和
g
。
fclose(fp);
关于c - 如何在C中读取/解析输入?常见问题,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47419842/
我是一名优秀的程序员,十分优秀!