cscanffflush

How do I limit the input of scanf to integers and floats (numbers in general)


I am curently learning C Programming in University and I got the task to write a program witch puts out the interception points with 0 / x-axis of any function ax^2+bx+c.

In order to make sure that the input is correct (int, float, etc.) I run the following while loop. Prior to that a is definded as a double.

printf("Input for a=");

while (scanf("%lf", &a) == 0)
{
    fflush(stdin);
    scanf("%lf", &a);
    printf("Incorrect Input! New Input needed.\n"); 
    printf("a=");
}

I am aware that the fflush(stdin) operator only clears the buffer when a second input function occurs and therefore the fflush inside the loop does not clear the buffer and therefore the condition of the loop is always true and thus I created an infinite loop.

My professor also forbids the use of goto, which is why I am here, because I can't come up with a reasonable solution that solves this problem. I also tried and failed with:

do
{
    printf("\nInput for a= ");
    scanf("%lf", &a);
}

while (isdigit(a));
{
    printf("Thank you.\n");
}

With this arrangement I get the failure notification: Expression c >= -1 && <= 255. I guess this has to do with false variable definition (double, digit) or such.

However my original question was whether there is an elegant solution to this problem or at least any solution.


Solution

  • Lukas, I'm still not 100% clear and your:

    "Im am asking how to distinguish between all numbers and every other possible input for scanf."

    scanf can provide conversion to a single given type based on the conversion specifier used, so you can't read both int and float with the same scanf statement and a single conversion specifier. (now you can read a line with, e.g. fgets and then use alternate sscanf statements and check the remaining characters to do that)

    That said, I think I understand what you are asking and can answer both the fflush and read of a value questions.

    To begin, when using scanf you must check the return and handle three-cases. (1) EOF, the user cancels input with a Ctrl+d (or Ctrl+z on windows); (2) a matching or input failure occurs resulting in a return of less than the number of conversion specifiers used; and finally (3) the good input case (where you impose any additional checks you have, e.g. positive, less than 100, etc..)

    While fflush(stdin) is undefined behavior on most systems, there are a number of implementations that allow it. (primarily windows, but Linux allows it for seekable stream, e.g. a file redirected on stdin) Bottom line, it isn't portable without caveats, so it's best to provide a simple equivalent with getchar(), e.g.

    void empty_stdin (void)
    {
        int c = getchar();
    
        while (c != '\n' && c != EOF)
            c = getchar();
    }
    

    As for scanf, as I mentioned in the comment, it is far easier to enclose the scanf call within an infinite loop which you only break when the input satisfies all your constraints. A simple example requiring integer input would be:

    int getint (int *value, const char *prompt)
    {
        /* loop continually until good input or canceled */
        for (;;) {
            int rtn;
            fputs (prompt, stdout);     /* display prompt */
            rtn = scanf ("%d", value);
    
            if (rtn == EOF) {   /* user generated manual EOF */
                fputs ("<user canceled input>\n", stderr);
                return 0;
            }
            empty_stdin();  /* all other cases - empty input buffer */
            if (rtn == 1)   /* good input, break */
                break;
            /* otherwise matching failure */
            fputs ("  error: invalid integer input.\n", stderr);
        }
        return *value;  /* value also availale through pointer */
    }
    

    Putting it altogether in a simple example, you would have:

    #include <stdio.h>
    
    void empty_stdin (void)
    {
        int c = getchar();
    
        while (c != '\n' && c != EOF)
            c = getchar();
    }
    
    int getint (int *value, const char *prompt)
    {
        /* loop continually until good input or canceled */
        for (;;) {
            int rtn;
            fputs (prompt, stdout);     /* display prompt */
            rtn = scanf ("%d", value);
    
            if (rtn == EOF) {   /* user generated manual EOF */
                fputs ("<user canceled input>\n", stderr);
                return 0;
            }
            empty_stdin();  /* all other cases - empty input buffer */
            if (rtn == 1)   /* good input, break */
                break;
            /* otherwise matching failure */
            fputs ("  error: invalid integer input.\n", stderr);
        }
        return *value;  /* value also availale through pointer */
    }
    
    int main (void) {
    
        int v,
            i = getint (&v, "enter integer value: ");
    
        if (i)
            printf ("\ninteger: %d\n", v);
    
        return 0;
    }
    

    Example Use/Output

    Where you can now do your best to break any input routine you write. If you find a problem, go fix it and try to break it again.

    The code above allows for fairly robust input of any one given type of value, e.g.

    $ ./bin/scanfint
    enter integer value: no
      error: invalid integer input.
    enter integer value: apples, banannas, and pears
      error: invalid integer input.
    enter integer value: 21
    
    integer: 21
    

    Look things over and let me know if your question was slightly different, or if you have further questions about the answer.