问题描述与截图
在初学java的过程中难免从键盘获取数据,一旦从键盘获取数据,那么就离不开一个类那就是Scanner类,在此类中可以获得键盘的输入流,从而获取输入数据
但是在此过程中当创建了Scanner类的时候会eclipse会报一个警告
提醒我们没有将Scanner类的对象回收。
在回收进行的时候一旦失手进行了==两次close()==方法的调用就会报一个这样的错误java.util.NoSuchElementException
问题简易代码
package com.qianfeng.day12;
import java.util.Scanner;
public class ScannerCloseTest {
public static void main(String[] args) {
Scanner input = new Scanner(System.in);
String str = input.next();
System.out.println("第一次输入的内容为:" + str);
input.close();
testClose();
}
public static void testClose() {
Scanner input = new Scanner(System.in);
String str1 = input.next();
System.out.println("第二次输入的内容为:" + str1);
input.close();
input.close();
}
}
Scanner原码分析
通过Scanner类从键盘输入的大致的步骤为
- 导入java.util.Scanner的包
- 创建对象:Scanner input = new Scanner(System.in);
- 输入数据:String str = input.next();
- 回收对象与输入流:input.close();
在第二步,左边为创建一个对象的引用,而右边则调用了Scanner的这个构造器:
public Scanner(InputStream source) {
this(new InputStreamReader(source), WHITESPACE_PATTERN);
}
传入的参数为System.in:其中in为System中的静态属性InputStream的一个引用
//System类中的原码
public final static InputStream in = null;
//InputStream类为一个抽象类
public abstract class InputStream implements Closeable
在最后一步回收方法的原码为:
public void close() {
if (closed)
return;
if (source instanceof Closeable) {
try {
((Closeable)source).close();
} catch (IOException ioe) {
lastException = ioe;
}
}
sourceClosed = true;
source = null;
closed = true;
}
总体来说就是通过Scanner类传入了一个键盘的输入流,然后通过一次close方法操作,就将程序的输入流关闭了,就算再次创建一个Scanner类的对象,也改变不了这个输入流被关闭的现实。所以以后在需要输入的时候就会报没有输入流的错误
解决办法
- 忽视eclipse上的警告,不要调用close()方法
- 如果程序只在同一个类中,在所有方法的最后才进行回收,在所有输入操作结束后,不管回收多少次都没有问题。
- 如果程序是多个类的联系操作,可以考虑将Scanner对象设置为成员,而不是局部成员,这样eclipse会自动回收,也不会报警告