解决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
发表评论 取消回复