getchar does not stop when using scanf

前端 未结 5 894
傲寒
傲寒 2020-11-27 22:38

I have a difficulty understanding getchar(). In the following program getchar works as expected:

#include 


int mai         


        
相关标签:
5条回答
  • 2020-11-27 22:59

    Whitespace is a delimiter for 5y3 %s format specifier, and newline is regarded as whitespace, so it remains buffered. Console input is normally line oriented, so a subsequent call to getchar() will return immediately because a 'line' remains buffered.

    scanf("%s", command );
    while( getchar() != '\n' ){ /* flush to end of input line */ }
    

    Equally if you use getchar() or %c to get a single character you normally need to flush the line, but in this case the character entered may itself be a newline so you need a slightly different solution:

    scanf("%c", ch );
    while( ch != '\n' && getchar() != '\n' ){ /* flush to end of input line */ }
    

    similarly for getchar():

    ch = getchar();
    while( ch != '\n' && getchar() != '\n' ){ /* flush to end of input line */ }
    

    The sensible thing to do of course is to wrap these solutions into stand-alone specialised input functions that you can reuse and also use as a place to put common input validation and error checking code (as in Daniel Fischer's answer which sensibly checks for EOF - you would normally want to avoid having to duplicate those checks and error handling everywhere).

    0 讨论(0)
  • 2020-11-27 23:08

    Well, I have something easier: add another getchar() ... problem solved!!

    0 讨论(0)
  • 2020-11-27 23:09

    after taking command input flush the stdin.

    fflush(stdin);
    

    but flushing a input stream results in undefined behavior (though Microsoft's C library defines the behaviour as an extension).

    0 讨论(0)
  • 2020-11-27 23:11

    The input is only sent to the program after you typed a newline, but

    scanf("%s", command );
    

    leaves the newline in the input buffer, since the %s(1) format stops when the first whitespace character is encountered after some non-whitespace, getchar() then returns that newline immediately and doesn't need to wait for further input.

    Your workaround works because it clears the newline from the input buffer before calling getchar() once more.

    To emulate the behaviour, clear the input buffer before printing the message,

    scanf("%s", command);
    int c;
    do {
        c = getchar();
    }while(c != '\n' && c != EOF);
    if (c == EOF) {
        // input stream ended, do something about it, exit perhaps
    } else {
        printf("Type Enter to continue\n");
        getchar();
    }
    

    (1) Note that using %s in scanf is very unsafe, you should restrict the input to what your buffer can hold with a field-width, scanf("%99s", command) will read at most 99 (sizeof(command) - 1)) characters into command, leaving space for the 0-terminator.

    0 讨论(0)
  • 2020-11-27 23:15

    I'd rather first use fgets and then use sscanf to parse the input. I have been doing this stuff like this for a long time and the behaviour has been more predictable than using plain scanf.

    0 讨论(0)
提交回复
热议问题