解决System.in关闭后无法再继续使用流的问题
关于new BufferedReader(new InputStreamReader(System.in))
出现的Stream Closed
的错误
在某天练习io
流的时候无意中写了一个代码一直在报Stream Closed
错误
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.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循环中不便于观察所以重新写一段代码
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
进行搭配使用,代码如下:
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
参考连接
(93条消息) Java实验:解决System.in关闭后无法再继续使用流的问题_水木的编程那些事儿的博客-CSDN博客
System.in关闭问题-CSDN社区