我正在使用 Scanner
方法 nextInt()
及 nextLine()
用于读取输入。
看起来是这样的:
System.out.println("Enter numerical value");
int option;
option = input.nextInt(); // Read numerical value from input
System.out.println("Enter 1st string");
String string1 = input.nextLine(); // Read 1st string (this is skipped)
System.out.println("Enter 2nd string");
String string2 = input.nextLine(); // Read 2nd string (this appears right after reading numerical value)
问题是,输入数值后,第一个 input.nextLine()
被跳过,第二个 input.nextLine()
执行,因此我的输出如下所示:
Enter numerical value
3 // This is my input
Enter 1st string // The program is supposed to stop here and wait for my input, but is skipped
Enter 2nd string // ...and this line is executed and waits for my input
我测试了我的应用程序,看起来问题在于使用 input.nextInt()
. 如果我删除它,那么两个 string1 = input.nextLine()
及 string2 = input.nextLine()
按照我的要求执行。
20条答案
按热度按时间pcww981p1#
那是因为
Scanner.nextInt
方法不会读取通过单击“enter”创建的输入中的换行符,因此调用Scanner.nextLine
读取该换行符后返回。在使用时,您将遇到类似的行为
Scanner.nextLine
之后Scanner.next()
或任何Scanner.nextFoo
方法(除nextLine
本身)。解决方法:
要么放一个
Scanner.nextLine
每次之后都打电话Scanner.nextInt
或Scanner.nextFoo
使用该行的其余部分,包括换行符或者,更好的是,通读输入
Scanner.nextLine
并将输入转换为所需的正确格式。例如,您可以使用Integer.parseInt(String)
方法。fquxozlt2#
问题在于input.nextint()方法-它只读取int值。因此,当您继续使用input.nextline()读取时,您将收到“\n”enter键。因此,要跳过此操作,必须添加input.nextline()。希望这一点现在已经清楚了。
试着这样做:
nlejzf6q3#
这是因为当你输入一个数字,然后按enter键,
input.nextInt()
只消耗数字,而不是“行尾”。什么时候input.nextLine()
执行时,它将使用第一个输入中仍在缓冲区中的“行尾”。相反,使用
input.nextLine()
紧接着input.nextInt()
dzhpxtsq4#
关于这个问题,我们似乎有很多疑问
java.util.Scanner
. 我认为更具可读性/惯用的解决方案是scanner.skip("[\r\n]+")
调用后删除任何换行符nextInt()
.编辑:正如@patrickparker在下面提到的,如果用户在数字后面输入任何空格,这将导致无限循环。请参阅他们的答案,以获得更好的模式,以便与skip一起使用:https://stackoverflow.com/a/42471816/143585
zf9nrax15#
它这样做是因为
input.nextInt();
没有捕捉到新线。您可以通过添加input.nextLine();
在下面或者,您可以采用c#风格,将下一行解析为整数,如下所示:
这样做同样有效,并且可以为您节省一行代码。
yacmzcpb6#
tl;博士
使用
scanner.skip("\\R")
(自skip
在何处使用正则表达式\R
表示每行之前的行分隔符)scanner.newLine()
调用,该调用在以下时间之后执行:scanner.next()
scanner.next*TYPE*()
方法,比如scanner.nextInt()
.或更安全的变体:
scanner.skip("\\R?")
在每个之前scanner.nextLine()
如果你不确定之后是否会调用它scanner.next()
或scanner.next*TypeName*()
.?
将使行分隔符序列可选(这将防止skip
(a)中的方法等待匹配序列-如果数据源仍然打开,如System.in
(b) 投掷java.util.NoSuchElementException
如果是终止/结束的数据源(如文件或字符串)你需要知道的事情:
表示几行的文本在行之间还包含不可打印的字符(我们称之为行分隔符),如
回车符(cr)-表示为
"\r"
)换行符(lf)-表示为
"\n"
)当您从控制台读取数据时,它允许用户键入他的响应,当他完成时,他需要以某种方式确认这一事实。为此,用户需要按键盘上的“回车”/“回车”键。
重要的是,除了确保将用户数据放置到标准输入(由
System.in
这是由Scanner
)还发送依赖操作系统的行分隔符(如windows)\r\n
)在它之后。因此,当您要求用户提供以下价值时:
age
,用户键入42并按enter键,标准输入将包含"42\r\n"
.问题
Scanner#nextInt
(及其他Scanner#nextType
方法)不允许扫描仪使用这些行分隔符。它将从中读取它们System.in
(否则扫描器怎么知道用户没有更多代表age
这将从标准输入中删除它们,但也会在内部缓存这些行分隔符。我们需要记住的是,所有的scanner方法总是从缓存的文本开始扫描。现在
Scanner#nextLine()
只需收集并返回所有字符,直到找到行分隔符(或流结束)。但由于从控制台读取数字后,行分隔符会立即在扫描仪的缓存中找到,因此它返回空字符串,这意味着扫描仪无法在这些行分隔符(或流结束)之前找到任何字符。顺便提一下
nextLine
也会消耗这些行分隔符。解决方案
所以,当您想要请求数字,然后请求整行,同时避免由于
nextLine
任何一个使用左边的行分隔符
nextInt
从扫描仪缓存使命感
nextLine
,或者我更容易理解的方法是打电话
skip("\\R")
或skip("\r\n|\r|\n")
让扫描仪跳过线分隔符匹配的部分(有关\R
: https://stackoverflow.com/a/31060125)不要使用
nextInt
(也不是next
,或任何nextTYPE
方法)。而是使用nextLine
并将每行中的数字(假设一行只包含一个数字)解析为适当的类型,如int
通过Integer.parseInt
.顺便说一句:
Scanner#nextType
方法可以跳过分隔符(默认情况下,所有空格,如制表符、行分隔符),包括扫描程序缓存的分隔符,直到找到下一个非分隔符值(标记)。多亏了这样的投入"42\r\n\r\n321\r\n\r\n\r\nfoobar"
代码将能够正确分配
num1=42
num2=321name=foobar
.k5ifujac7#
而不是
input.nextLine()
使用input.next()
,这应该可以解决问题。修改代码:
6fe3ivhb8#
如果要同时读取字符串和整数,解决方案是使用两个扫描仪:
holgip5t9#
如果您想快速扫描输入而不被scanner类nextline()方法所迷惑,请使用自定义输入扫描仪。
代码:
优点:
比bufferreader更快地扫描输入
减少时间复杂性
为下一个输入刷新缓冲区
方法:
scanchar()-扫描单个字符
scanint()-扫描整数值
scanlong()-扫描长值
scanstring()-扫描字符串值
scandouble()-扫描双值
scanint(int[]数组)-扫描整个数组(整数)
scanlong(long[]数组)-扫描整个数组(long)
用法:
将给定代码复制到java代码下面。
初始化给定类的对象
ScanReader sc = new ScanReader(System.in);
3.导入必要的类:import java.io.BufferedInputStream; import java.io.IOException; import java.io.InputStream;
4.从主方法抛出ioexception以处理异常5.使用提供的方法。6.享受例如:
ghg1uchk10#
为了避免该问题,请使用
nextLine();
紧接着nextInt();
因为它有助于清除缓冲区。当你按下ENTER
这个nextInt();
不捕获新行,因此跳过Scanner
稍后再编码。piah890a11#
sc.nextLine()
与解析输入相比,它的性能更好。因为就性能而言,这将是好的。ncecgwcz12#
我想我参加晚会已经很晚了。。
如前所述
input.nextLine()
得到int值后,将解决您的问题。您的代码不起作用的原因是,您的输入(输入int的地方)中没有其他内容可存储string1
. 我只想对整个主题多说一点。将NestRoin()视为扫描器类中的NeXFoT()方法中的奇数。让我们举一个很快的例子。假设我们有两行代码,如下所示:
如果我们输入以下值(作为一行输入)
54 234
我们的价值
firstNumber
及secondNumber
变量分别变为54和234。这种方式工作的原因是,当nextint()方法接受值时,不会自动生成新行馈送(即\n)。它只需要“next int”并继续。除了nextline()之外,其余的nextfoo()方法也是如此。nextline()在获取值后立即生成一个新行提要;这就是@rohitjain所说的新行提要被“消耗”的意思。
最后,next()方法只获取最近的字符串,而不生成新行;这使得它成为在同一行中获取单独字符串的首选方法。
我希望这有帮助。。快乐的编码!
smtd7mpg13#
wztqucjr14#
如果我期望一个非空的输入
避免:
–如果以下输入被未经检查的
scan.nextLine()
作为解决办法–仅部分读取行导致数据丢失,因为
scan.nextLine()
被替换为scan.next()
(输入:“Yeppie ya yeah”)–
Exception
使用解析输入时抛出的Scanner
方法(先读取,后解析)在上述示例中使用:
iqih9akk15#
使用两个扫描仪对象,而不是一个