I have a variable int number and I need code that only allows introducing 1, 2, 3, 4 or 5 to the variable.
I have scanf and I just do an if...else to check out if the number is between 1 and 5, but the point is that I don't know what to do if I introduce a character, e.g. 'q'. I mean, I want the program to say "That's a character, not a number."
use sscanf on a buffer (or in the example below, argv the program argument) to attempt to find a digit, then a character.
The return code from sscanf is :
...the number of input items successfully matched
and assigned, which can be fewer than provided for, or even zero in the
event of an early matching failure.
Here is an example program
#include <stdlib.h>
#include <stdio.h>
#include <errno.h>
#include <string.h>
int main(int argc, char ** argv){
int val=-1;
char other;
int code;
if (argc !=2 ) {
printf("please give an arg to program\n");
exit(0);
}
code=sscanf(argv[1],"%d",&val);
if (code == 1) {
printf( "number %d", val);
}
else {
code=sscanf(argv[1],"%c",&other);
if (code == 1) {
printf("character is %c", other);
}
else {
printf("error occured");
}
}
}
Only accept input of 1 to 5, inclusive:
#include <stdio.h>
int main() {
int rc, answer = 0;
printf("A number between 1 and 5: ");
rc = scanf("%i", &answer);
if(rc == EOF || rc == 0 || answer > 5 || answer <= 0) {
fprintf(stderr, "Only numbers between 1 and 5, please.\n");
return -99;
} else {
printf("You entered: %d\n", answer);
return 0;
}
}
Return Value
These functions return the number of input items successfully matched and assigned, which can be fewer than provided for, or even zero in the event of an early matching failure.
The value EOF is returned if the end of input is reached before either the first successful conversion or a matching failure occurs. EOF is also returned if a read error occurs, in which case the error indicator for the stream (see ferror(3)) is set, and errno is set indicate the error.
Basically, scanf will continue to scan until a match is made or it fails. An example program would be this:
#include <stdio.h>
int main() {
int number, result;
result = scanf("%i", &number);
if(result == EOF) {
// an error occured.
fprintf(stderr, "That's not a number");
} else {
// logic! yay!
// the result is in `number`.
}
}
Use man scanf for more information.
Try with this code:
#include <stdio.h>
int clearStdin()
{
while (getchar() != '\n');
return 1;
}
int main()
{
int integer = 0;
char character;
printf("Enter an integer not a character:\n");
do
{
printf("? ");
} while (((scanf("%d%c", &integer, &character) != 2) || (character != '\n')) && clearStdin());
printf("You entered the following integer: %d\n", integer);
return 0;
}
scanf will return the number of successful conversions and assignments, so you can check that to make sure you read at least 1 digit character:
int value;
...
if ( scanf( "%d", &value ) == 1 )
{
// read at least 1 digit
}
else
{
// first character was not whitespace or a digit
}
What's this "read at least 1 digit" nonsense? The %d conversion specifier tells scanf to skip any leading whitespace, then read and convert decimal digits until it finds a non-digit character. This means if you type a string like "12w4", scanf will convert and assign 12 to value, return 1, and leave "w4" in the input stream to foul up the next read.
That's probably not the behavior you want. You could scan for a following non-digit character, like so:
#include <ctype.h>
...
int value;
char follow;
...
if ( scanf( "%d%c", &value, &follow ) == 2 )
{
if ( !isspace( follow ))
{
fprintf( stderr, "detected non-digit character in input\n" );
}
}
else
{
// first character was not whitespace or a digit
}
You've caught that case, but your input may now be in a bad state; how do you know if the next character was part of a bad input or not? IOW, how do you know the input was 12w4 and not 12w 4?
Personally, I take a different approach. I read the input as a string, then use strtol to perform the conversion:
/**
* Assuming a 64-bit integer value, average 3.2 bits per decimal digit, we
* need to store up to 22 digits plus an optional sign character plus the 0
* terminator
*/
define INPUT_SIZE 24
...
char input[INPUT_SIZE] = {0};
long value = 0;
...
if ( fgets( input, sizeof input, stdin ))
{
char *newline = strchr( input, '\n' );
if ( !newline )
{
fprintf( stderr, "Input is too long for a 64-bit integer...rejecting\n" );
/**
* Cycle through the input stream until we see the newline character
*/
while ( fgets( input, sizeof input, stdin ) && !strchr( input, '\n' ))
;
}
else
{
char *chk; // will point to the first character that's *not*
// part of a decimal integer
long temp = strtol( input, &chk, 10 );
if ( !isspace( *chk ) && *chk != 0 )
{
fprintf( stderr, "%s is not a valid integer input\n", input );
}
else
{
value = temp;
}
}
}
This allows me to catch cases where the input value is obviously too long to store in the target data type, something from which scanf does not protect you. It also allows me to catch cases where a non-digit character is fat-fingered in the middle of the input and reject the whole input outright, something else from which scanf doesn't protect you.
Simple read 1 char and test it.
int ch;
while ((ch = fgetc(stdin)) != EOF) {
if ((ch < '1') || (ch > '5')) {
fputs(stdout, "That's a not number 1 to 5.\n");
}
else {
ch -= '0';
printf("Good: %d entered\n", ch);
}
}
Related
The while loop below breaks when a letter is added. I would like it to continue to loop even if a letter is added, as well as if a negative number is added.
Here's my current code:
float monthpay;
printf("Please enter your monthly wage:\t");
scanf("%f", &monthpay);
while (monthpay <= 0)
{
printf("\nThat was invalid");
printf("\n Please try again: ");
scanf("%f", &monthpay);
}
printf("%.2f", monthpay);
I was trying to make a loop that would loop if a letter was added by mistake.
scanf() returns number of input items that were successfully matched and assigned. If you don't check the return value (rv) your variable (monthpay) may be uninitialized, and it's undefined behavior to use it.
void flush() {
for(;;) {
char ch;
int rv = scanf("%c", &ch);
if(rv == EOF || rv == 1 && ch == '\n') return;
}
}
printf("Please enter your monthly wage:\t");
float monthpay;
for(;;) {
int rv = scanf("%f", &monthpay);
if(rv == EOF)
return;
if(rv == 1)
break;
printf("\nThat was invalid");
printf("\n Please try again: ");
flush();
}
printf("%.2f", monthpay);
After trying to scan a float, scan and discard whitespace except for newline, scan discard and count any remaining characters that are not a newline, scan a newline.
The %n specifier will provide a count of the characters processed by the scan up to the specifier.
The asterisk in the scanset %*[ \t\f\v] tells scanf to discard the matching characters. The carat in the scanset %*[^\n] tells scanf to process the non-matching characters so everything that is not a newline will be discarded. %1[\n] tells scanf to scan one character and it must be a newline.
If a float is entered followed only by whitespace, the loop will exit as scanned will be 2 and extra will be 0.
#include <stdio.h>
int main ( void) {
char newline[2] = "";
int extra = 0;
int scanned = 0;
float monthpay = 0.0;
do {
printf ( "enter monthly pay\n");
scanned = scanf ( "%f", &monthpay);
if ( scanned == EOF) {
return 1;
}
extra = 0;
scanf ( "%*[ \f\t\v]"); // scan and discard whitespace except newline
scanf ( "%*[^\n]%n", &extra); // scan discard and count non-whitespace
scanned += scanf ( "%1[\n]", newline);
} while ( monthpay < 0 || scanned != 2 || extra != 0);
return 0;
}
You should always check the return value of scanf to determine whether the input was successfully converted, before you attempt to use the result.
However, for line-based user input, using the function scanf is generally not advisable, because its behavior is not intuitive. It usually does not consume an entire line of input, which can cause all kinds of trouble. For example, scanf leaving the newline character on the input stream can cause this problem. If the user enters 6abc\n, then the situation is even worse: In that case, scanf will match the 6 as valid input, but leave abc\n on the input stream, which will most likely cause the next input operation to not behave as intended, unless you explicitly discard the remainder of the line from the input stream beforehand.
For the reasons stated above, it is generally better to always read an entire line of input at once as a string, for example using the function fgets. You can then use the function strtof to attempt to convert the string to a number.
Here is an example, which performs extensive input validation:
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <ctype.h>
#include <errno.h>
float get_float_from_user( const char *prompt );
int main( void )
{
float monthpay;
//loop forever until user enters a valid number
for (;;) //infinite loop, equivalent to while(1)
{
monthpay = get_float_from_user( "Please enter your monthly wage: ");
if ( monthpay > 0 )
{
//input is ok, so we can break out of the loop
break;
}
printf( "Please enter a positive number!\n" );
}
printf( "Input was valid. You entered: %.2f\n", monthpay );
return 0;
}
float get_float_from_user( const char *prompt )
{
//loop forever until user enters a valid number
for (;;)
{
char buffer[1024], *p;
float f;
//prompt user for input
fputs( prompt, stdout );
//get one line of input from input stream
if ( fgets( buffer, sizeof buffer, stdin ) == NULL )
{
fprintf( stderr, "Unrecoverable input error!\n" );
exit( EXIT_FAILURE );
}
//make sure that entire line was read in (i.e. that
//the buffer was not too small)
if ( strchr( buffer, '\n' ) == NULL && !feof( stdin ) )
{
int c;
printf( "Line input was too long!\n" );
//discard remainder of line
do
{
c = getchar();
if ( c == EOF )
{
fprintf( stderr, "Unrecoverable error reading from input!\n" );
exit( EXIT_FAILURE );
}
} while ( c != '\n' );
continue;
}
//attempt to convert string to number
errno = 0;
f = strtof( buffer, &p );
if ( p == buffer )
{
printf( "Error converting string to number!\n" );
continue;
}
//make sure that number is representable as a "float"
if ( errno == ERANGE )
{
printf( "Number out of range error!\n" );
continue;
}
//make sure that remainder of line contains only whitespace,
//so that input such as "6abc" gets rejected
for ( ; *p != '\0'; p++ )
{
if ( !isspace( (unsigned char)*p ) )
{
printf( "Unexpected input encountered!\n" );
//cannot use `continue` here, because that would go to
//the next iteration of the innermost loop, but we
//want to go to the next iteration of the outer loop
goto continue_outer_loop;
}
}
return f;
continue_outer_loop:
continue;
}
}
This program has the following behavior:
Please enter your monthly wage: abc
Error converting string to number!
Please enter your monthly wage: 6abc
Unexpected input encountered!
Please enter your monthly wage: 6e+2000
Number out of range error!
Please enter your monthly wage: -5
Please enter a positive number!
Please enter your monthly wage: 0
Please enter a positive number!
Please enter your monthly wage: 6.3
Input was valid. You entered: 6.30
The function get_float_from_user that I am using in the code above is based on the function get_int_from_user from this answer of mine to another question. See that answer for a further explanation on how that function works.
Basically, I need to ensure that input is an integer, like so:
do {
printf("Enter > ");
scanf("%d", &integer);
} while (/* user entered a char instead of an int */);
I have tried various methods, but it always end up with run-time error or infinite loop when I tried to enter a char. I have knew that fflush(stdin) is an undefined behavior, which is better not to involve it in my code in order to prevent any error plus it no longer works in VS2015 due to some reasons.
The codes below are the method that I have tried:
typedef enum {false, true} bool;
int ipt;
char c;
bool wrong_ipt;
do {
c = '\0';
printf("Enter > ");
scanf("%d%c", &ipt, &c); //infinite loop occurs while a char has been entered
} while (c != '\n');
do {
c = '\0';
printf("Enter > ");
} while (scanf("%d", &ipt) != EOF);
do {
wrong_ipt = false;
do {
ipt = NULL;
printf("Enter > ");
scanf("%d", &ipt);
if (ipt == NULL) {
wrong_ipt = true;
break;
}
} while (ipt == NULL);
} while (wrong_ipt);
Is there anyway other than fflush(stdin) which can be used to prevent the infinite loop when user entered a char in C?
Thank you
The problem is that "scanf()" can leave unread data in your input buffer. Hence the "infinite loop".
Another issue is that you should validate the return value from scanf(). If you expect one integer value ... and scanf returns "0" items read ... then you know something went wrong.
Here is an example:
#include <stdio.h>
void discard_junk ()
{
int c;
while((c = getchar()) != '\n' && c != EOF)
;
}
int main (int argc, char *argv[])
{
int integer, i;
do {
printf("Enter > ");
i = scanf("%d", &integer);
if (i == 1) {
printf ("Good value: %d\n", integer);
}
else {
printf ("BAD VALUE, i=%i!\n", i);
discard_junk ();
}
} while (i != 1);
return 0;
}
Sample output:
Enter > A
BAD VALUE, i=0!
Enter > B
BAD VALUE, i=0!
Enter > 1
Good value: 1
'Hope that helps!
This is a very good example of why scanf should generally not be used for user input.
Since user input is line-based, one would expect that an input function would always read one line of input at a time. However, that is not the way that the function scanf behaves. Instead, it consumes only as many characters as are required to match the %d conversion format specifier. If scanf is unable to match anything, then it does not consume any characters at all, so that the next call to scanf will fail for exactly the same reason (assuming that the same conversion specifier is used and that the invalid input is not explicitly discarded by you). This is what is happening in your code.
At the time of this writing, three of the other answers solve this problem by checking the return value of scanf and explicitly discarding the invalid input. However, all three (!) of these answers have the problem that they for example accept "6sdfj23jlj" as valid input for the number 6, although the whole line of input should obviously be rejected in this case. This is because scanf, as previously mentioned, does not read one line of input at a time.
Therefore, the best solution to your problem would probably be to use line-based input using fgets instead. That way, you will always read exactly one line of input at a time (assuming that the input buffer is large enough to store an entire line of input). After reading the line, you can then attempt to convert it to a number using strtol. Even if the conversion fails, the line of input will have been consumed from the input stream, so you will not have most of the problems described above.
A simple solution using fgets could look like this:
#include <stdio.h>
#include <stdlib.h>
#include <ctype.h>
int main( void )
{
char line[100];
long number;
//retry until user enters valid input
for (;;) //infinite loop, equivalent to while(1)
{
char *p;
//prompt user for input
printf( "Please enter a number: " );
//attempt to read one line of input
if ( fgets( line, sizeof line, stdin ) == NULL )
{
fprintf( stderr, "Unrecoverable input error!\n" );
exit( EXIT_FAILURE );
}
//attempt to convert input to number
number = strtol( line, &p, 10 );
//verify that conversion was successful
if ( p == line )
{
printf( "Invalid input!\n" );
continue;
}
//verify that remainder of line only contains
//whitespace, so that input such as "6sdfj23jlj"
//gets rejected
for ( ; *p != '\0'; p++ )
{
if ( !isspace( (unsigned char)*p ) )
{
printf( "Encountered invalid character!\n" );
//cannot use `continue` here, because that would go to
//the next iteration of the innermost loop, but we
//want to go to the next iteration of the outer loop
goto continue_outer_loop;
}
}
//input was valid, so break out of infinite loop
break;
//label for breaking out of nested loop
continue_outer_loop:
continue;
}
printf( "Input was valid.\n" );
printf( "The number is: %ld\n", number );
return 0;
}
Note that the goto statement should generally not be used. However, in this case, it is necessary, in order to break out of a nested loop.
This program has the following output:
Please enter a number: 94hjj
Encountered invalid character!
Please enter a number: 5455g
Encountered invalid character!
Please enter a number: hkh7
Invalid input!
Please enter a number: 6sdfj23jlj
Encountered invalid character!
Please enter a number: 67
Input was valid.
The number is: 67
However, this code is still not perfect. It still has the following problems:
If the user enters 100 characters of input in a single line, then the entire line won't fit into the input buffer. In that case, two calls to fgets will be required to read the entire line, and the program will incorrectly treat that line as two separate lines of input.
The code does not check if the number the user entered is representable as a long (e.g. whether the number is excessively large). The function strtol reports this by setting errno accordingly (which is a feature that scanf lacks).
These two problems can be fixed too, by performing additional checks and error handling. However, the code is now becoming so complex that it makes sense to put it all into its own function:
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <ctype.h>
#include <limits.h>
#include <errno.h>
int get_int_from_user( const char *prompt )
{
//loop forever until user enters a valid number
for (;;)
{
char buffer[1024], *p;
long l;
//prompt user for input
fputs( prompt, stdout );
//get one line of input from input stream
if ( fgets( buffer, sizeof buffer, stdin ) == NULL )
{
fprintf( stderr, "Unrecoverable input error!\n" );
exit( EXIT_FAILURE );
}
//make sure that entire line was read in (i.e. that
//the buffer was not too small)
if ( strchr( buffer, '\n' ) == NULL && !feof( stdin ) )
{
int c;
printf( "Line input was too long!\n" );
//discard remainder of line
do
{
c = getchar();
if ( c == EOF )
{
fprintf( stderr, "Unrecoverable error reading from input!\n" );
exit( EXIT_FAILURE );
}
} while ( c != '\n' );
continue;
}
//attempt to convert string to number
errno = 0;
l = strtol( buffer, &p, 10 );
if ( p == buffer )
{
printf( "Error converting string to number!\n" );
continue;
}
//make sure that number is representable as an "int"
if ( errno == ERANGE || l < INT_MIN || l > INT_MAX )
{
printf( "Number out of range error!\n" );
continue;
}
//make sure that remainder of line contains only whitespace,
//so that input such as "6sdfj23jlj" gets rejected
for ( ; *p != '\0'; p++ )
{
if ( !isspace( (unsigned char)*p ) )
{
printf( "Unexpected input encountered!\n" );
//cannot use `continue` here, because that would go to
//the next iteration of the innermost loop, but we
//want to go to the next iteration of the outer loop
goto continue_outer_loop;
}
}
return l;
continue_outer_loop:
continue;
}
}
int main( void )
{
int number;
number = get_int_from_user( "Please enter a number: " );
printf( "Input was valid.\n" );
printf( "The number is: %d\n", number );
return 0;
}
Your fundamental mistake is that you never tell your program to consume the invalid input. In words, you told the program:
Read an integer if there is one. (Otherwise read nothing)
If you didn't get an integer, go back to step 1.
I assume what you thought you were doing was
Read the input, and if it was an integer, store it.
If it wasn't an integer, go back to step 1.
So what you need to do is to rewrite your code so it does what you meant to do (or come up with some other approach, as in the other answer). That is, write your program to:
Read some amount of input. (e.g. a line)
Scan the input to see if it is an integer, and store it.
If it wasn't an integer, go back to step 1.
Some relevant functions you may find useful are fgets, sscanf, and atoi. (also, try to resist the temptation to write buggy code; e.g. if you intend to read a line of input, make sure you actually do so, and correctly. Many people are lazy and will do the wrong thing if the line is long; e.g. just read part of the line, or cause a buffer overflow)
The format specifier %d tells to scanf that an integer value is expected in the command line. When the user enters a line of data, it's read as a string, and then scanf attempts to understand if the entered string can be interpreted as the decimal digits of an integer number.
If this interpretation succeds, then the value thus found is stored in the integer variable you passed as argument.
The number of rigth replacements done by scanf are retrieved by that function in the form of an int value. Since you expect only one input, the value 1 will indicates that everything was fine.
So, if there was an error, as for example, the user entered a non-valid integer number, the number returnd by scanf is less than the number of format specifiers. In this case, a value less than 1 informs that an error happened:
int ipt, succeded;
do {
printf("ipt? ");
succeded = scanf("%d", &ipt);
if (succeded < 1) { // Clean the input
while (getchar() != '\n')
;
}
} while(succeded < 1);
I know I'm a little late to answer this but you could reduce the use of loops and use jump statements instead and then just use one if statement to check for bad input and still achieve the same result
enter: printf("Enter > ");
i = scanf("%d", &integer);
if (i != 1) {
printf ("BAD VALUE, i=%i!\n", i);
discard_junk ();
goto enter:
}
printf ("Good value: %d\n", integer);
How do I make an error message if only enter or a space is entered?
do
{
printf("Please enter a username: ");
scanf("%s", name);
if(name[0] == '\n')
{
printf("invalid input");
}
printf("> %s\n", name);
}
while(strlen(name) < 2 || strlen(name) > 15);
The newline will result in an empty string so the format specifier is not satisfied, so scanf() does not return. It is somewhat arcane behaviour, but:
int count = scanf( "%16[^ \n]s", name ) ;
will return when newline or leading whitespace is entered with count == 0. It will also accept no more than 16 characters, preventing a buffer overrun while allowing the > 15 characters test.
To avoid unnecessary tests and multiple calls to strlen() for the same string:
#include <stdio.h>
#include <string.h>
#include <ctype.h>
#include <stdbool.h>
int main()
{
char name[20] = "" ;
bool name_valid = false ;
while( !name_valid )
{
printf("Please enter a username: ");
int count = scanf("%16[^ \n]", name);
size_t len = count == 0 ? 0 : strlen( name ) ;
name_valid = len > 1 && len < 16 ;
if( !name_valid )
{
int ch ;
do
{
ch = getchar() ;
} while( ch != '\n' && ch != EOF ) ;
printf("invalid input\n");
}
}
printf("> %s\n", name);
return 0;
}
Note that in "invalid input" you need to remove the buffered newline and any preceding junk.
I personally don't like using scanf() b/c it results in a lot of trouble and it is not safe to use it in a production environment. I recommend using fgets() instead, and here is how it works:
#include <stdio.h>
#include <string.h>
int main()
{
char buffer[50];
int buffer_size = sizeof(buffer);
/* accepts string input from user */
printf("Please enter a username: ");
fgets(buffer, buffer_size, stdin);
/* then remove "\n" from the string
A newline character makes fgets stop reading, but it is considered a valid character by the function and included in the string copied to str.
*/
buffer[strcspn(buffer, "\n")] = 0;
/* check if user has entered an empty string */
/* strcmp(): compares two strings and returns 0 if comparision is true */
if ((strcmp(buffer, "")) == 0 || (strcmp(buffer, " ")) == 0)
printf("I can detect that you printed nothing! \n");
else
printf("You printed: %s", buffer);
return 0;
}
It may seem like an absurd amount of work to do just to get input from user, but this is how it works in C.
Our task for school lesson is to input a number of integers. We don't know how many there will be.
I would like to know if there is a possibility to format scanf function that either stores integer or terminate itself by pressing enter.
Can I somehow put together scanf("%d") which only stores integers and scanf("%[^\n]) which terminates scanf function?
What I have known yet is that I cannot use scanf("%d%[^\n]) because scanf is waiting for that one integer, which I don't want to input because I already stored all integers I had to.
I don't really like a possibility to store string of all those integers into an array and then convert it to elements of another array with the exact numbers.
The scanf function is difficult to use correctly.
Instead, read a line at a time with fgets. If the entered string is just a newline, you exit the loop. If not, use strtol to parse the value. You'll know if just an integer was entered if the end pointer points to the newline at the end of the input.
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>
int main()
{
char line[100], *p;
long val;
while (fgets(line, sizeof(line), stdin)) {
// if enter was pressed by itself, quit loop
if (!strcmp(line, "\n")) {
break;
}
errno = 0;
val = strtol(line, &p, 10);
if (errno) {
perror("error reading value");
} else if ((p != line) && (*p == '\n')) {
// a valid integer was read
printf("read value %ld\n", val);
} else {
// a non-integer was read or extra characters were entered
printf("not a valid integer: %s", line);
}
}
return 0;
}
Scan a character. Skip space and tab. Exit on newline.
Unget most recent character and try to scan an integer. If unable to scan an integer, scan and discard non-digit except space tab and newline.
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
int main ( void) {
char i = '\0';
int value = 0;
int result = 0;
printf ( "type number separated by space then enter\n");
while ( 1 == scanf("%c",&i)) {//scan a character
if ( ' ' == i || '\t' == i) {
continue;
}
if ( i == '\n') {
break;//newline so leave loop
}
ungetc ( i, stdin);//replace the character in input stream
if ( 1 == ( result = scanf ( "%d", &value))) {
printf ( " number entered as %d\n", value);
}
else {
scanf ( "%*[^0-9 \t\n]");//clean non digits except space tab newline
//or you could just break;
}
}
return 0;
}
Basically, I need to ensure that input is an integer, like so:
do {
printf("Enter > ");
scanf("%d", &integer);
} while (/* user entered a char instead of an int */);
I have tried various methods, but it always end up with run-time error or infinite loop when I tried to enter a char. I have knew that fflush(stdin) is an undefined behavior, which is better not to involve it in my code in order to prevent any error plus it no longer works in VS2015 due to some reasons.
The codes below are the method that I have tried:
typedef enum {false, true} bool;
int ipt;
char c;
bool wrong_ipt;
do {
c = '\0';
printf("Enter > ");
scanf("%d%c", &ipt, &c); //infinite loop occurs while a char has been entered
} while (c != '\n');
do {
c = '\0';
printf("Enter > ");
} while (scanf("%d", &ipt) != EOF);
do {
wrong_ipt = false;
do {
ipt = NULL;
printf("Enter > ");
scanf("%d", &ipt);
if (ipt == NULL) {
wrong_ipt = true;
break;
}
} while (ipt == NULL);
} while (wrong_ipt);
Is there anyway other than fflush(stdin) which can be used to prevent the infinite loop when user entered a char in C?
Thank you
The problem is that "scanf()" can leave unread data in your input buffer. Hence the "infinite loop".
Another issue is that you should validate the return value from scanf(). If you expect one integer value ... and scanf returns "0" items read ... then you know something went wrong.
Here is an example:
#include <stdio.h>
void discard_junk ()
{
int c;
while((c = getchar()) != '\n' && c != EOF)
;
}
int main (int argc, char *argv[])
{
int integer, i;
do {
printf("Enter > ");
i = scanf("%d", &integer);
if (i == 1) {
printf ("Good value: %d\n", integer);
}
else {
printf ("BAD VALUE, i=%i!\n", i);
discard_junk ();
}
} while (i != 1);
return 0;
}
Sample output:
Enter > A
BAD VALUE, i=0!
Enter > B
BAD VALUE, i=0!
Enter > 1
Good value: 1
'Hope that helps!
This is a very good example of why scanf should generally not be used for user input.
Since user input is line-based, one would expect that an input function would always read one line of input at a time. However, that is not the way that the function scanf behaves. Instead, it consumes only as many characters as are required to match the %d conversion format specifier. If scanf is unable to match anything, then it does not consume any characters at all, so that the next call to scanf will fail for exactly the same reason (assuming that the same conversion specifier is used and that the invalid input is not explicitly discarded by you). This is what is happening in your code.
At the time of this writing, three of the other answers solve this problem by checking the return value of scanf and explicitly discarding the invalid input. However, all three (!) of these answers have the problem that they for example accept "6sdfj23jlj" as valid input for the number 6, although the whole line of input should obviously be rejected in this case. This is because scanf, as previously mentioned, does not read one line of input at a time.
Therefore, the best solution to your problem would probably be to use line-based input using fgets instead. That way, you will always read exactly one line of input at a time (assuming that the input buffer is large enough to store an entire line of input). After reading the line, you can then attempt to convert it to a number using strtol. Even if the conversion fails, the line of input will have been consumed from the input stream, so you will not have most of the problems described above.
A simple solution using fgets could look like this:
#include <stdio.h>
#include <stdlib.h>
#include <ctype.h>
int main( void )
{
char line[100];
long number;
//retry until user enters valid input
for (;;) //infinite loop, equivalent to while(1)
{
char *p;
//prompt user for input
printf( "Please enter a number: " );
//attempt to read one line of input
if ( fgets( line, sizeof line, stdin ) == NULL )
{
fprintf( stderr, "Unrecoverable input error!\n" );
exit( EXIT_FAILURE );
}
//attempt to convert input to number
number = strtol( line, &p, 10 );
//verify that conversion was successful
if ( p == line )
{
printf( "Invalid input!\n" );
continue;
}
//verify that remainder of line only contains
//whitespace, so that input such as "6sdfj23jlj"
//gets rejected
for ( ; *p != '\0'; p++ )
{
if ( !isspace( (unsigned char)*p ) )
{
printf( "Encountered invalid character!\n" );
//cannot use `continue` here, because that would go to
//the next iteration of the innermost loop, but we
//want to go to the next iteration of the outer loop
goto continue_outer_loop;
}
}
//input was valid, so break out of infinite loop
break;
//label for breaking out of nested loop
continue_outer_loop:
continue;
}
printf( "Input was valid.\n" );
printf( "The number is: %ld\n", number );
return 0;
}
Note that the goto statement should generally not be used. However, in this case, it is necessary, in order to break out of a nested loop.
This program has the following output:
Please enter a number: 94hjj
Encountered invalid character!
Please enter a number: 5455g
Encountered invalid character!
Please enter a number: hkh7
Invalid input!
Please enter a number: 6sdfj23jlj
Encountered invalid character!
Please enter a number: 67
Input was valid.
The number is: 67
However, this code is still not perfect. It still has the following problems:
If the user enters 100 characters of input in a single line, then the entire line won't fit into the input buffer. In that case, two calls to fgets will be required to read the entire line, and the program will incorrectly treat that line as two separate lines of input.
The code does not check if the number the user entered is representable as a long (e.g. whether the number is excessively large). The function strtol reports this by setting errno accordingly (which is a feature that scanf lacks).
These two problems can be fixed too, by performing additional checks and error handling. However, the code is now becoming so complex that it makes sense to put it all into its own function:
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <ctype.h>
#include <limits.h>
#include <errno.h>
int get_int_from_user( const char *prompt )
{
//loop forever until user enters a valid number
for (;;)
{
char buffer[1024], *p;
long l;
//prompt user for input
fputs( prompt, stdout );
//get one line of input from input stream
if ( fgets( buffer, sizeof buffer, stdin ) == NULL )
{
fprintf( stderr, "Unrecoverable input error!\n" );
exit( EXIT_FAILURE );
}
//make sure that entire line was read in (i.e. that
//the buffer was not too small)
if ( strchr( buffer, '\n' ) == NULL && !feof( stdin ) )
{
int c;
printf( "Line input was too long!\n" );
//discard remainder of line
do
{
c = getchar();
if ( c == EOF )
{
fprintf( stderr, "Unrecoverable error reading from input!\n" );
exit( EXIT_FAILURE );
}
} while ( c != '\n' );
continue;
}
//attempt to convert string to number
errno = 0;
l = strtol( buffer, &p, 10 );
if ( p == buffer )
{
printf( "Error converting string to number!\n" );
continue;
}
//make sure that number is representable as an "int"
if ( errno == ERANGE || l < INT_MIN || l > INT_MAX )
{
printf( "Number out of range error!\n" );
continue;
}
//make sure that remainder of line contains only whitespace,
//so that input such as "6sdfj23jlj" gets rejected
for ( ; *p != '\0'; p++ )
{
if ( !isspace( (unsigned char)*p ) )
{
printf( "Unexpected input encountered!\n" );
//cannot use `continue` here, because that would go to
//the next iteration of the innermost loop, but we
//want to go to the next iteration of the outer loop
goto continue_outer_loop;
}
}
return l;
continue_outer_loop:
continue;
}
}
int main( void )
{
int number;
number = get_int_from_user( "Please enter a number: " );
printf( "Input was valid.\n" );
printf( "The number is: %d\n", number );
return 0;
}
Your fundamental mistake is that you never tell your program to consume the invalid input. In words, you told the program:
Read an integer if there is one. (Otherwise read nothing)
If you didn't get an integer, go back to step 1.
I assume what you thought you were doing was
Read the input, and if it was an integer, store it.
If it wasn't an integer, go back to step 1.
So what you need to do is to rewrite your code so it does what you meant to do (or come up with some other approach, as in the other answer). That is, write your program to:
Read some amount of input. (e.g. a line)
Scan the input to see if it is an integer, and store it.
If it wasn't an integer, go back to step 1.
Some relevant functions you may find useful are fgets, sscanf, and atoi. (also, try to resist the temptation to write buggy code; e.g. if you intend to read a line of input, make sure you actually do so, and correctly. Many people are lazy and will do the wrong thing if the line is long; e.g. just read part of the line, or cause a buffer overflow)
The format specifier %d tells to scanf that an integer value is expected in the command line. When the user enters a line of data, it's read as a string, and then scanf attempts to understand if the entered string can be interpreted as the decimal digits of an integer number.
If this interpretation succeds, then the value thus found is stored in the integer variable you passed as argument.
The number of rigth replacements done by scanf are retrieved by that function in the form of an int value. Since you expect only one input, the value 1 will indicates that everything was fine.
So, if there was an error, as for example, the user entered a non-valid integer number, the number returnd by scanf is less than the number of format specifiers. In this case, a value less than 1 informs that an error happened:
int ipt, succeded;
do {
printf("ipt? ");
succeded = scanf("%d", &ipt);
if (succeded < 1) { // Clean the input
while (getchar() != '\n')
;
}
} while(succeded < 1);
I know I'm a little late to answer this but you could reduce the use of loops and use jump statements instead and then just use one if statement to check for bad input and still achieve the same result
enter: printf("Enter > ");
i = scanf("%d", &integer);
if (i != 1) {
printf ("BAD VALUE, i=%i!\n", i);
discard_junk ();
goto enter:
}
printf ("Good value: %d\n", integer);