代码的目的是查看字符串的存储方式(以\n
还是\0
结尾)。
void print(char*);
int main(void) {
//fscanf block
char *ss;
ss = malloc(sizeof(char)*100);
printf("String for fscanf: ");
fscanf(stdin, "%s", ss);
printf("For fscanf:\n");
print(ss);
//fgets block
char *gs;
gs = malloc(sizeof(char)*100);
printf("String for fgets: ");
fgets(gs, 10, stdin);
printf("For fgets:\n");
print(gs);
free(ss);
free(gs);
}
void print(char* str) {
int done = 0;
for (int i=0; i<100; i++){
if (done == 3) {
break;
}
if (str[i] == '\0') {
printf("Zero: %d\n", i);
done++;
}
if (str[i] == '\n') {
printf("Return: %d\n", i);
done++;
}
}
}
但是出于某种原因,当我首先放入fscanf块时,它不会停止并为fgets字符串req,它只是跳过并输出:
String for fscanf: Hello
For fscanf:
Zero: 5
Zero: 6
String for fgets: For fgets:
Return: 0
Zero: 1
但是当我把fgets块放到第一位时它工作正常:
String for fgets: Hello
For fgets:
Return: 5
Zero: 6
String for fscanf: Hello
For fscanf:
Zero: 5
Zero: 6
为什么会这样?
答案 0 :(得分:3)
scanf
和fscanf
有很多警告they get their own page in the C FAQ。您mixing fscanf and fgets is covered by 12.18a的具体问题。
简而言之,您键入了Hello\n
。 fscanf(stdin, "%s", ss)
正在读取字符串Hello
,但不会读取换行符。那还在stdin
。 fgets
然后读取换行符并停止。在调试这些类型的东西时,最好在调试输出周围加上引号,以便您可以看到这些内容。 printf("'%s'", gs);
一般情况下,请勿混用fscanf
和fgets
。坚持一个或另一个。通常避免scanf
和fscanf
来读取和解析行,将读取和解析结合在一起会导致很多问题。相反,使用fgets
来读取整行,sscanf
来解析它。
答案 1 :(得分:1)
它没有按预期工作,因为当您执行fscanf(stdin, "%s", ss);
并按下ENTER
时,它会将ENTER
留在第二个缓冲区中。请注意,ENTER
是有效字符。因此,它将读取\n
作为第二个输入,它不会等待fgets(gs, 10, stdin);
。为此,您需要在此
stdin
缓冲区
fscanf(stdin, "%s", ss);
完成任务的一种方法是使用getchar()
。例如
//fscanf block
getchar();/* help to clear the stdin buffer */
//fgets block
另请阅读此处为何不应使用scanf()
&amp;应该使用什么而不是scanf()
http://c-faq.com/stdio/scanfprobs.html