How do I keep a Scanner from throwing exceptions when the wrong type is entered?
You can use one of the many hasNext*
methods that Scanner
has for pre-validation.
if (in.hasNextInt()) {
int a = in.nextInt() ;
System.out.println(a);
} else {
System.out.println("Sorry, couldn't understand you!");
}
This prevents InputMismatchException
from even being thrown, because you always make sure that it WILL match before you read it.
java.util.Scanner API
-
boolean hasNextInt()
: Returnstrue
if the next token in this scanner's input can be interpreted as an int value in the default radix using thenextInt()
method. The scanner does not advance past any input. -
String nextLine()
: Advances this scanner past the current line and returns the input that was skipped.
Do keep in mind the sections in bold. hasNextInt()
doesn't advance past any input. If it returns true
, you can advance the scanner by calling nextInt()
, which will not throw an InputMismatchException
.
If it returns false
, then you need to skip past the "garbage". The easiest way to do this is just by calling nextLine()
, probably twice but at least once.
Why you may need to do nextLine()
twice is the following: suppose this is the input entered:
42[enter]
too many![enter]
0[enter]
Let's say the scanner is at the beginning of that input.
-
hasNextInt()
is true,nextInt()
returns42
; scanner is now at just before the first[enter]
. -
hasNextInt()
is false,nextLine()
returns an empty string, a secondnextLine()
returns"too many!"
; scanner is now at just after the second[enter]
. -
hasNextInt()
is true,nextInt()
returns0
; scanner is now at just before the third[enter]
.
Here's an example of putting some of these things together. You can experiment with it to study how Scanner
works.
Scanner in = new Scanner (System.in) ;
System.out.println("Age?");
while (!in.hasNextInt()) {
in.next(); // What happens if you use nextLine() instead?
}
int age = in.nextInt();
in.nextLine(); // What happens if you remove this statement?
System.out.println("Name?");
String name = in.nextLine();
System.out.format("[%s] is %d years old", name, age);
Let's say the input is:
He is probably close to 100 now...[enter]
Elvis, of course[enter]
Then the last line of the output is:
[Elvis, of course] is 100 years old
In general I really, really dislike using the same library call for both reading and parsing. Language libraries seem to be very inflexible and often just can't be bent to your will.
The first step that pulls data from System.in should not be able to fail, so have it read it as a string into a variable, then convert that string variable to an int. If the conversion fails, great--print your error and continue.
When you wrap your stream with something that can throw an exception, it gets kind of confusing just what state the whole mess leaves your stream in.
It's always a benefit to have your application throw an error when an error occurs opposed to ways to keep it from happening.
One alternative is to wrap the code inside a try {...}
catch {...}
block for InputMismatchException
.
You might also want to wrap the code inside a while
loop to have the Scanner
keep prompting until a specific condition is met.