利用Scanner类从输入流中获取数据以及next()、nextXxx()、nextLine()方法辨析

本文介绍了如何使用Java的Scanner类从输入流中获取数据,特别是处理double值的场景。通过实例展示了如何结束输入流,强调了Ctrl+Z在Windows系统中的作用。接着详细比较了next()、nextXxx()和nextLine()方法的差异,指出next()和nextXxx()以空格和换行作为分隔符,而nextLine()仅识别回车作为结束。此外,还讨论了不同结束符对hasNext()和hasNextLine()判断的影响。

问题来源:

算法第四版第1.1节习题32题,标准输入流中有一系列double值,要获取作处理。那么,怎么从输入流中获取数据呢?

Scanner类从键盘获取数据;

Scanner是一个基于正则表达式的文本扫描器,他可以从文件、输入流、字符串中解析出基本数据类型和字符串值,十分方便。下文中我们没有用到正则表达式。

从输入流中获取double值存入list数组留待取用:

	   Scanner scanner = new Scanner(System.in);		
		ArrayList<Double> list = new ArrayList<Double>();
		while (scanner.hasNextDouble()) {
			list.add(scanner.nextDouble());
		}

在执行过程中遇到了问题——怎么结束输入流?当我按下回车的时候,并不能结束输入,方法仍然在等待:(注意右上角小红点)


显然这是一个阻塞式方法,那么我们就要告知scanner,输入结束了。

在控制台内,我们可以按下Ctrl+Z,结束输入流。在进行第一次尝试的时候,我输入完“1 2 3”,直接按下Ctrl+Z,结果输入是结束了,但scanner并没有获取到数据:


打印出了一个空数组。然后我发现在控制台里输入的数据仍然可以修改,怀疑控制台显示的数据并没有送给输入流,于是在输入完数据后按下enter键,再按下Ctrl+Z,成功获得了数据:



也可以设置标志让方法结束:

示例:当输入-1时结束获取

		ArrayList<Double> list = new ArrayList<Double>();
		double temp ;
		while ((temp= scanner.nextDouble()) != -1) {
			list.add(temp);
		}
		System.out.println(list.toString());



探究Scanner的next(),nextXxx()和nextLine()方法的区别:

默认情况下,Scanner以空格和换行(空格键、Tab键、Enter键)作为分隔符

1. next()方法k可以获取任意有效字符。next()方法以及形如nextXxx()的方法会忽略有效字符前的空格和回车,以空格和换行作为结束符,读取到有效字符后的空格和换行结束,这就决定了它们不能得到带有空格的字符串。next()的返回值类型是String,nextXxx()的返回值是相应的基本数据类型。

		int i = 0;
		String temp;
		while(scanner.hasNext()){
			temp = scanner.next();
			i++;
			System.out.println("获取的第"+i+"个值是:"+temp);
		}

需要按Ctrl+Z来结束输入。

2. nextXxx()方法的功能就是将输入的字符串转化为相应的基本数据类型,如果不能转换就抛出java.util.InputMismatchException异常.

		System.out.println(scanner.nextDouble());
		System.out.println(scanner.nextBoolean());
		System.out.println(scanner.nextInt());
		System.out.println(scanner.nextInt());


3.  nextLine()方法只识别Enter键作为结束,也就是说它获取回车键前的所有字符,包括空格。nextLine()方法的返回值类型是String,可以得到带空格的字符串。

		int i = 0;
		String temp;
		while(scanner.hasNext()){
			temp = scanner.nextLine();
			i++;
			System.out.println("获取的第"+i+"个值是:"+temp);
		}

需要按Ctrl+Z来结束输入。

4. 由于next(),nextXxx() 和 nextLine() 的结束符不同,所以在混合使用时,要注意处理,防止出错。

情形一:

		while(scanner .hasNext()){	
			System.out.println(scanner.nextInt());
			System.out.println(scanner.nextInt());
			System.out.println(scanner.nextInt());
			System.out.println(scanner.nextLine());
		}

   注意:nextXxx()过滤掉了前面的空格;nextLine()读取到了前面遗留下来的空格。

需要按Ctrl+Z来结束输入。

情形二:

		FileReader fr = new FileReader("1.txt");
		Scanner scanner = new Scanner(fr);
		System.out.println(scanner.nextInt());
		System.out.println(scanner.nextInt());
		System.out.println(scanner.nextInt());
		System.out.println(scanner.nextLine());

1.txt:


注意:这里没有打印出Hello World! ,却打印出了一个空字符串(注意光标位置),是为什么呢?这个文本在计算机里是这么存储的:1 2 3\r\nHello World!    当3被读取掉的时候回车键还在,所以需要把缓存里这个回车键读掉才能读到后面的内容,也就是下一行内容:

		FileReader fr = new FileReader("1.txt");
		Scanner scanner = new Scanner(fr);
		System.out.println(scanner.nextInt());
		System.out.println(scanner.nextInt());
		System.out.println(scanner.nextInt());
		scanner.nextLine();
		System.out.println(scanner.nextLine());

情形三:结束符不同,hasNext()方法和hasNextLine()的判断依据也会不同。

		FileReader fr = new FileReader("2.txt");
		Scanner scanner = new Scanner(fr);
		while(scanner.hasNext()){
			System.out.println(scanner.next());
			System.out.println(scanner.hasNext());
			System.out.println(scanner.hasNextLine());
		

2.txt: 注意光标在字符3之后


输出结果:


修改一下2.txt:只是在文件结尾敲了一个回车键:注意光标位置在第二行开头


重新运行相同的程序:


最后一行,hasNextLIne()显示true,因为它检测到后面还有一个回车换行符。

最后贴一个stackoverflow上看到的这两个方法判断的依据:

hasNextLine() checks to see if there is another LinePattern in the buffer.

hasNext() checks to see if there is a parseable token in the buffer, as separated by the scanner's delimiter.

private static Pattern WHITESPACE_PATTERN = Pattern.compile(
                                            "\\p{javaWhitespace}+");

private static final String LINE_SEPARATOR_PATTERN =
                                       "\r\n|[\n\r\u2028\u2029\u0085]";
private static final String LINE_PATTERN = ".*("+LINE_SEPARATOR_PATTERN+")|.+$";

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值