C scanf() and fgets() problem

2019-02-07 11:58发布

I'm trying to read user input and store it as a string including the whitespace. I did a search for a solution and was pointed to fgets() or scanf(%[^\n], str). But both these solutions give me an error.

This is what I have:

//MAX_CHARACTERS is set to 30

 scanf("%d", &input);
 if (input == 1){
        int pr;
        char na[MAX_CHARACTERS+1];
        printf("\nEnter the name: ");
        scanf("%[^\t\n]", &na);
        while (strlen(na)>MAX_CHARACTERS){
            printf("\nName is too long, enter new name: ");
            scanf("%[^\t\n]", &na);
        }// end na check
        printf("\nEnter priority: ");
        scanf("%d", &pr);
        while (pr>MAX_PRIORITY || pr <MIN_PRIORITY){
            printf("\nBad priority, enter new priority (0-100): ");
            scanf("%d", &pr);
        }//end pr check

It works fine if I use %s in all instances of %[^\t\n] but when I use %[^\t\n] or

fgets(na, 30, stdin), it skips the first scanf for name and goes straight to "Enter priority: ". Then when I print, I have a blank name with whatever priority I entered.

EDIT: Sorry, the missing quotes on the first scanf is a typo. Not a cause of the problem. I typed in '1' for the first scanf("%d", input).

FIXED IT

Since it won't let me post an answer yet,

Someone figured it out. Incase anyone's still interested, the problem was the first scanf().

scanf("%d", &input);

It is leaving a \n in the buffer. The second one is taking the \n and reading it as an input so it gets skipped.

SOLUTION:

Putting a

fflush(stdin); //right after the if statement seems to have fixed the issue. 

Thanks for everyone's help.

标签: c scanf fgets
4条回答
女痞
2楼-- · 2019-02-07 12:36
    scanf("%d",&choice);
    fgetc(stdin);

what i did is to read the fgetc from stdin and this leave the '\n' so your buffer is crear.

查看更多
【Aperson】
3楼-- · 2019-02-07 12:37
scanf("%[^\t\n]", &na);

should be:

scanf("%[^\t\n]", na);

ie., remove the &

Also fgets should work fine except that it will include the newline from when the user presses enter. You will have to strip it off.

查看更多
对你真心纯属浪费
4楼-- · 2019-02-07 12:40

the first line of the code:

scanf(%d, &input);

should be :

scanf("%d", &input);

i.e. the quotes are missing

查看更多
一夜七次
5楼-- · 2019-02-07 12:43

This is the problem with using scanf() to take in user input, it's not made to handle strings or any bad inputs. scanf() is meant to read formatted input (e.g., files with a strict and specific format). When taking in input from the user, input could be malformed and causes problems. You should use fgets() to read the input and parse it yourself.

To answer why the second scanf() call gets skipped, it's because the first call only consumes the digits that gets inputted. However the newline from pressing Enter remains in the input buffer and is in turn, the input of the second call. To fix this, you'd want to remove that newline character from the input buffers. The safest way to do that would be to use getchar() until the newline character is read, then continue with your program. And avoid using the fflush(stdin) pattern, that's not safe and can cause more problems than it's worth.

Here's a simple function you can use to read characters until a newline is read:

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

Call this after any call to scanf() that you're processing non-string data.

查看更多
登录 后发表回答