为什么不打电话给fgets?

时间:2018-06-13 02:18:05

标签: c string scanf fgets

代码的目的是查看字符串的存储方式(以\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

为什么会这样?

2 个答案:

答案 0 :(得分:3)

scanffscanf有很多警告they get their own page in the C FAQ。您mixing fscanf and fgets is covered by 12.18a的具体问题。

简而言之,您键入了Hello\nfscanf(stdin, "%s", ss)正在读取字符串Hello,但不会读取换行符。那还在stdinfgets然后读取换行符并停止。在调试这些类型的东西时,最好在调试输出周围加上引号,以便您可以看到这些内容。 printf("'%s'", gs);

一般情况下,请勿混用fscanffgets。坚持一个或另一个。通常避免scanffscanf来读取和解析行,将读取和解析结合在一起会导致很多问题。相反,使用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