首页 > 代码库 > nextLine()紧跟在nextXXX()后面读取不到数据的问题,以及两者的小结

nextLine()紧跟在nextXXX()后面读取不到数据的问题,以及两者的小结

     用Java在在线oj平台上做题时,要求的标准输入一般是“Scanner cin=new Scanner(System.in);”。

    其实还有更有效率的方法:Scanner cin=new Scanner(new BufferedInputStream(System.in))。笔者更常用这种方法,效率更高。但是由于本文主题不在于此,因此不再赘述。

    回到正题,之前就发现nextXXX()(包括next()方法)后面紧跟nextLine()的话,后者不会读取到数据,输出的话为空字符串。这两天做了几个题,又涉及到这个问题了。既然开博了,就把这个问题分析、总结,记录下来,以后忘了的话翻翻博客就记起来了。

    写了一段代码,测试所有的cin.nextXXX()方法。

    代码如下:

技术分享
  1 import java.math.BigDecimal;  2 import java.math.BigInteger;  3 import java.util.Scanner;  4   5 /**  6  *  测试:1.所有的nextXXX()方法(包括next()),后面直接跟nextLine()读取数据;  7  *  2.先加一句nextLine(),再用nextLine()读取数据,两者的区别。  8  *  将每个代码块中间的注释"//cin.nextLine()",去掉"//",即可实现第2种方法。  9  *  @author Hubert 10  * 11  */ 12 public class ScannerTest { 13  14     public static void main(String[] args) { 15         // TODO Auto-generated method stub 16     Scanner cin=new Scanner(System.in); 17      18     //测试cin.next() 19     System.out.println("测试 cin.next()后面紧跟着nextLine()读取数据:"); 20     System.out.println("请输入一个字符串:"); 21     String next=cin.next(); 22     //cin.nextLine(); 23     System.out.println("请输入一个字符串:"); 24     String nextLine1=cin.nextLine(); 25     System.out.println("next()方法读取的值为:"+next); 26     System.out.println("nextLine()方法读取的值为:"+nextLine1+"\n"); 27      28     //测试cin.nextInt() 29     System.out.println("测试cin.nextInt()后面紧跟着nextLine()读取数据:"); 30     System.out.println("请输入一个整数:"); 31     int nextInt=cin.nextInt(); 32     //cin.nextLine(); 33     System.out.println("请输入一个字符串:"); 34     String nextLine2=cin.nextLine(); 35     System.out.println("nextInt()方法读取的值为:"+nextInt); 36     System.out.println("nextLine()方法读取的值为:"+nextLine2+"\n"); 37      38     //测试cin.nextDouble() 39     System.out.println("测试cin.nextDouble()后面紧跟着nextLine()读取数据:"); 40     System.out.println("请输入一个小数:"); 41     double nextDouble=cin.nextDouble(); 42     //cin.nextLine(); 43     System.out.println("请输入一个字符串:"); 44     String nextLine3=cin.nextLine(); 45     System.out.println("nextDouble()方法读取的值为:"+nextDouble); 46     System.out.println("nextLine()方法读取的值为:"+nextLine3+"\n"); 47      48     //测试cin.nextFloat() 49     System.out.println("测试cin.nextFloat()后面紧跟着nextLine()读取数据:"); 50     System.out.println("请输入一个小数:"); 51     float nextFloat=cin.nextFloat(); 52     //cin.nextLine(); 53     System.out.println("请输入一个字符串:"); 54     String nextLine4=cin.nextLine(); 55     System.out.println("nextFloat()方法读取的值为:"+nextFloat); 56     System.out.println("nextLine()方法读取的值为:"+nextLine4+"\n"); 57      58     //测试cin.nextBoolean() 59     System.out.println("测试cin.nextBoolean()后面紧跟着nextLine()读取数据:"); 60     System.out.println("请输入一个布尔值(true或者false):"); 61     boolean nextBoolean=cin.nextBoolean(); 62     //cin.nextLine(); 63     System.out.println("请输入一个字符串:"); 64     String nextLine5=cin.nextLine(); 65     System.out.println("nextBoolean()方法读取的值为:"+nextBoolean); 66     System.out.println("nextLine()方法读取的值为:"+nextLine5+"\n"); 67      68     //测试cin.nextLong() 69     System.out.println("测试cin.nextLong()后面紧跟着nextLine()读取数据:"); 70     System.out.println("请输入一个整数:"); 71     long nextLong=cin.nextLong(); 72     //cin.nextLine(); 73     System.out.println("请输入一个字符串:"); 74     String nextLine6=cin.nextLine(); 75     System.out.println("nextLong()方法读取的值为:"+nextLong); 76     System.out.println("nextLine()方法读取的值为:"+nextLine6+"\n"); 77      78     //测试cin.nextByte() 79     System.out.println("测试cin.nextByte()后面紧跟着nextLine()读取数据:"); 80     System.out.println("请输入一个整数(-128~127):"); 81     byte nextByte=cin.nextByte(); 82     //cin.nextLine(); 83     System.out.println("请输入一个字符串:"); 84     String nextLine7=cin.nextLine(); 85     System.out.println("nextByte()方法读取的值为:"+nextByte); 86     System.out.println("nextLine()方法读取的值为:"+nextLine7+"\n"); 87      88     //测试cin.nextShort() 89     System.out.println("测试cin.nextShort()后面紧跟着nextLine()读取数据:"); 90     System.out.println("请输入一个整数:"); 91     short nextShort=cin.nextShort(); 92     //cin.nextLine(); 93     System.out.println("请输入一个字符串:"); 94     String nextLine8=cin.nextLine(); 95     System.out.println("nextShort()方法读取的值为:"+nextShort); 96     System.out.println("nextLine()方法读取的值为:"+nextLine8+"\n"); 97      98     //测试cin.nextBigInteger() 99     System.out.println("测试cin.nextBigInteger()后面紧跟着nextLine()读取数据:");100     System.out.println("请输入一个整数:");101     BigInteger nextBigInteger=cin.nextBigInteger();//扫描下一个输入作为BigInteger(可以实现任意大的大数)102     //cin.nextLine();103     System.out.println("请输入一个字符串:");104     String nextLine9=cin.nextLine();105     System.out.println("nextBigInteger()方法读取的值为::"+nextBigInteger);106     System.out.println("nextLine()方法读取的值为:"+nextLine9+"\n");107     108     //测试cin.nextBigDecimal()109     System.out.println("测试cin.nextBigDecimal()后面紧跟着nextLine()读取数据:");110     System.out.println("请输入一个小数:");111     BigDecimal nextBigDecimal=cin.nextBigDecimal();//扫描下一个输入作为BigDecimal(可以实现任意精度)112     //cin.nextLine();113     System.out.println("请输入一个字符串:");114     String nextLine10=cin.nextLine();115     System.out.println("nextBigDecimal()方法读取的值为:"+nextBigDecimal);116     System.out.println("nextLine()方法读取的值为:"+nextLine10+"\n");117     118     cin.close();119     } 120 }
ScannerTest

    根据提示,输入"test,2,3.0,4.0,true,5,6,7,8,9,10.0",输出结果如下:

技术分享
 1 测试 cin.next()后面紧跟着nextLine()读取数据: 2 请输入一个字符串: 3 test 4 请输入一个字符串: 5 next()方法读取的值为:test 6 nextLine()方法读取的值为: 7  8 测试cin.nextInt()后面紧跟着nextLine()读取数据: 9 请输入一个整数:10 211 请输入一个字符串:12 nextInt()方法读取的值为:213 nextLine()方法读取的值为:14 15 测试cin.nextDouble()后面紧跟着nextLine()读取数据:16 请输入一个小数:17 3.018 请输入一个字符串:19 nextDouble()方法读取的值为:3.020 nextLine()方法读取的值为:21 22 测试cin.nextFloat()后面紧跟着nextLine()读取数据:23 请输入一个小数:24 4.025 请输入一个字符串:26 nextFloat()方法读取的值为:4.027 nextLine()方法读取的值为:28 29 测试cin.nextBoolean()后面紧跟着nextLine()读取数据:30 请输入一个布尔值(true或者false):31 true32 请输入一个字符串:33 nextBoolean()方法读取的值为:true34 nextLine()方法读取的值为:35 36 测试cin.nextLong()后面紧跟着nextLine()读取数据:37 请输入一个整数:38 639 请输入一个字符串:40 nextLong()方法读取的值为:641 nextLine()方法读取的值为:42 43 测试cin.nextByte()后面紧跟着nextLine()读取数据:44 请输入一个整数(-128~127):45 746 请输入一个字符串:47 nextByte()方法读取的值为:748 nextLine()方法读取的值为:49 50 测试cin.nextShort()后面紧跟着nextLine()读取数据:51 请输入一个整数:52 853 请输入一个字符串:54 nextShort()方法读取的值为:855 nextLine()方法读取的值为:56 57 测试cin.nextBigInteger()后面紧跟着nextLine()读取数据:58 请输入一个整数:59 960 请输入一个字符串:61 nextBigInteger()方法读取的值为::962 nextLine()方法读取的值为:63 64 测试cin.nextBigDecimal()后面紧跟着nextLine()读取数据:65 请输入一个小数:66 10.067 请输入一个字符串:68 nextBigDecimal()方法读取的值为:10.069 nextLine()方法读取的值为:
输出结果1

    发现所有的cin.nextXXX()方法,后面紧跟cin.nextLine()读取数据的话,cin.nextLine()是读取不到数据的,在结果中可以看到每个cin.nextLine()读取到的都是空字符串。

    查看了一下Javadoc中nextLine()方法的说明:“Advances this scanner past the current line and returns the input that was skipped. This method returns the rest of the current line, excluding any line separator at the end. The position is set to the beginning of the next line.”。nextLine()官方说明是跳过当前行,并且返回被跳过的输入。这个方法能够返回当前行剩余的内容,排除行尾的行分隔符在外。最终将光标设定在下一行的开始。该方法会从光标处开始扫描,直到遇到行分隔符(\r\n),然后返回扫描的内容,并将光标设置为下一行的开始。上述的测试中,因为在输入内容后,需要敲击回车才能使程序继续执行。此时相当于光标和行分隔符(回车)之间内容为"",因此自然cin.nextLine()读取到的是""了。

    解决方法就是在cin.nextXXX()方法后,在加一个cin.nextLine(),作用是跳过该行的剩余内容(即""),并将光标设置为下一行的开始。

    在上述代码中,将每个代码块中的"//cin.nextLine()"中的注释符号去掉,再次进行测试。根据提示,输入"test,line1,2,line2,3.0,line3,4.0,line4,true,line5,6,line6,7,line7,8,line8,9,line9,10.0,line10"。输出结果为:

技术分享
 1 测试 cin.next()后面紧跟着nextLine()读取数据: 2 请输入一个字符串: 3 test 4 请输入一个字符串: 5 line1 6 next()方法读取的值为:test 7 nextLine()方法读取的值为:line1 8  9 测试cin.nextInt()后面紧跟着nextLine()读取数据:10 请输入一个整数:11 212 请输入一个字符串:13 line214 nextInt()方法读取的值为:215 nextLine()方法读取的值为:line216 17 测试cin.nextDouble()后面紧跟着nextLine()读取数据:18 请输入一个小数:19 3.020 请输入一个字符串:21 line322 nextDouble()方法读取的值为:3.023 nextLine()方法读取的值为:line324 25 测试cin.nextFloat()后面紧跟着nextLine()读取数据:26 请输入一个小数:27 4.028 请输入一个字符串:29 line430 nextFloat()方法读取的值为:4.031 nextLine()方法读取的值为:line432 33 测试cin.nextBoolean()后面紧跟着nextLine()读取数据:34 请输入一个布尔值(true或者false):35 true36 请输入一个字符串:37 line538 nextBoolean()方法读取的值为:true39 nextLine()方法读取的值为:line540 41 测试cin.nextLong()后面紧跟着nextLine()读取数据:42 请输入一个整数:43 644 请输入一个字符串:45 line646 nextLong()方法读取的值为:647 nextLine()方法读取的值为:line648 49 测试cin.nextByte()后面紧跟着nextLine()读取数据:50 请输入一个整数(-128~127):51 752 请输入一个字符串:53 line754 nextByte()方法读取的值为:755 nextLine()方法读取的值为:line756 57 测试cin.nextShort()后面紧跟着nextLine()读取数据:58 请输入一个整数:59 860 请输入一个字符串:61 line862 nextShort()方法读取的值为:863 nextLine()方法读取的值为:line864 65 测试cin.nextBigInteger()后面紧跟着nextLine()读取数据:66 请输入一个整数:67 968 请输入一个字符串:69 line970 nextBigInteger()方法读取的值为::971 nextLine()方法读取的值为:line972 73 测试cin.nextBigDecimal()后面紧跟着nextLine()读取数据:74 请输入一个小数:75 10.076 请输入一个字符串:77 line1078 nextBigDecimal()方法读取的值为:10.079 nextLine()方法读取的值为:line10
输出结果2

    这次cin.nextLine()能够成功地读取数据了。
    看了一下next()方法的说明。“Finds and returns the next complete token from this scanner. A complete token is preceded and followed by input that matches the delimiter pattern.”。意思是找到并返回扫描到的完整的标记,而一个完整的标记处于分隔符之间(处于行首的标记,可以看做是在换行"\r\n"这个分隔符之后)。就是说读取并返回两个分隔符之间的标记(token)。比如说输入“     input     ",标记"input"被两个分隔符(空格)夹在中间,那么这个方法会读取并返回"input"。其他的nextXXX()方法也是这个原理。

  

     

     

nextLine()紧跟在nextXXX()后面读取不到数据的问题,以及两者的小结