解决System.in关闭后无法再继续使用流的问题
关于new BufferedReader(new InputStreamReader(System.in))出现的Stream Closed的错误
在某天练习io流的时候无意中写了一个代码一直在报Stream Closed错误
java复制代码package com.cyc;
import java.io.BufferedReader;
import java.io.File;
import java.io.IOException;
import java.io.InputStreamReader;
public class Test02 {
public static void main(String[] args) {
while (true){
BufferedReader br=new BufferedReader(new InputStreamReader(System.in));
try {
while (true){
String str=br.readLine();
System.out.println(str);
test.show();
br.close();
}
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
class test{
static void show(){
BufferedReader br=new BufferedReader(new InputStreamReader(System.in));
try {
br.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
报错部分
java复制代码java.io.IOException: Stream closed
at java.io.BufferedInputStream.getBufIfOpen(BufferedInputStream.java:170)
at java.io.BufferedInputStream.read(BufferedInputStream.java:336)
at sun.nio.cs.StreamDecoder.readBytes(StreamDecoder.java:284)
at sun.nio.cs.StreamDecoder.implRead(StreamDecoder.java:326)
at sun.nio.cs.StreamDecoder.read(StreamDecoder.java:178)
at java.io.InputStreamReader.read(InputStreamReader.java:184)
at java.io.BufferedReader.fill(BufferedReader.java:161)
at java.io.BufferedReader.readLine(BufferedReader.java:324)
at java.io.BufferedReader.readLine(BufferedReader.java:389)
at com.cyc.Test02.main(Test02.java:14)
由于写在while循环中不便于观察所以重新写一段代码
java复制代码public class Test01 {
public static void main(String[] args) throws IOException {
BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
String str = br.readLine();
System.out.println(str);
br.close();
BufferedReader bb = new BufferedReader(new InputStreamReader(System.in));
str = bb.readLine();
System.out.println(str);
bb.close();
}
}
使用debug模式查出在关闭流后再次重新new BufferedReader()时报错
无法重新打开System.in,System.out或System.err。底层本地流是连接到其他进程的文件描述符,或连接到应用程序无法识别的文件。一旦基础本地文件描述符关闭,就不可能重新打开它们。
上网查资料得知System.in是虚拟机在程序一开始的时候自动连接,断开之后就连接不上了所以每次关流之后重新开流就会报Stream Closed错误
以后一定在没有后续动作之后再关流
参考资料
问题简述
当使用BufferedReader从键盘读入数据时,若调用了close方法,再次使用BufferedReader从键盘读入数据的话便会报java.io.IOException: Stream closed异常,产生此异常的代码见文末介绍。
问题分析
在Java IO体系中,若关闭了最外层包装流,那么内层被包装的流也会被关闭,比如使用BufferedReader从键盘读入,代码为: BufferedReader reader = new BufferedReader(new InputStreamReader(System.in)); 读入键盘输入结束后调用reader.close()就会关闭BufferedReader、InputStreamReader和System.in三个流,此时若再使用BufferedReader 对System.in进行包装则会报java.io.IOException: Stream closed,这是因为调用reader.close()后又级联调用了System.in的close方法,因此System.in已经被关闭了,无法再次使用。
解决办法就是自己构造一个类来连接BufferedReader和System.in两个流,这个类需要是InputStreamReader的子类(因为要连接BufferedReader和System.in,此外在这个类中要重写(覆盖)close方法使其不调用System.in的close方法(将close设为空实现即可)。
解决方法(代码)
构造一个类使其继承InputStreamReader,然后用新类与BufferedReader进行搭配使用,代码如下:
java复制代码import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
class NotCloseISR extends InputStreamReader{
public NotCloseISR(InputStream in){
super(in);
}
public void close() throws IOException {
//do nothing
}
}
public class Test {
public static void main(String[] args) throws IOException {
testInput(); // 两次调用该函数,但每次调用后仅关闭BufferedReader的流而不关闭System.in流
testInput();
}
public static void testInput() throws IOException {
//对于传入的System.in做包装
BufferedReader reader = new BufferedReader(new NotCloseISR(System.in)); //
System.out.println("请输入一行字符:");
String str = reader.readLine();
System.out.println("输入的字符:" + str);
reader.close(); //
}
}
上述代码便不会关闭System.in流,可以多次调用testInput函数重复从键盘读入数据。 若上面代码: BufferedReader reader = new BufferedReader(new NotCloseISR(System.in)); 改为: BufferedReader reader = new BufferedReader(new InputStreamReader(System.in));, 即不使用自定义类NotCloseISR而使用InputStreamReader的话,那么当第二次调用testInput函数的时候便会报错,报错信息为:java.io.IOException: Stream closed
还没有评论,来说两句吧...