解决System.in关闭后无法再继续使用流的问题

关于new BufferedReader(new InputStreamReader(System.in))出现的Stream Closed的错误

在某天练习io流的时候无意中写了一个代码一直在报Stream Closed错误

  1. java复制代码package com.cyc;
  2. import java.io.BufferedReader;
  3. import java.io.File;
  4. import java.io.IOException;
  5. import java.io.InputStreamReader;
  6. public class Test02 {
  7. public static void main(String[] args) {
  8. while (true){
  9. BufferedReader br=new BufferedReader(new InputStreamReader(System.in));
  10. try {
  11. while (true){
  12. String str=br.readLine();
  13. System.out.println(str);
  14. test.show();
  15. br.close();
  16. }
  17. } catch (IOException e) {
  18. e.printStackTrace();
  19. }
  20. }
  21. }
  22. }
  23. class test{
  24. static void show(){
  25. BufferedReader br=new BufferedReader(new InputStreamReader(System.in));
  26. try {
  27. br.close();
  28. } catch (IOException e) {
  29. e.printStackTrace();
  30. }
  31. }
  32. }

报错部分

  1. java复制代码java.io.IOException: Stream closed
  2. at java.io.BufferedInputStream.getBufIfOpen(BufferedInputStream.java:170)
  3. at java.io.BufferedInputStream.read(BufferedInputStream.java:336)
  4. at sun.nio.cs.StreamDecoder.readBytes(StreamDecoder.java:284)
  5. at sun.nio.cs.StreamDecoder.implRead(StreamDecoder.java:326)
  6. at sun.nio.cs.StreamDecoder.read(StreamDecoder.java:178)
  7. at java.io.InputStreamReader.read(InputStreamReader.java:184)
  8. at java.io.BufferedReader.fill(BufferedReader.java:161)
  9. at java.io.BufferedReader.readLine(BufferedReader.java:324)
  10. at java.io.BufferedReader.readLine(BufferedReader.java:389)
  11. at com.cyc.Test02.main(Test02.java:14)

由于写在while循环中不便于观察所以重新写一段代码

  1. java复制代码public class Test01 {
  2. public static void main(String[] args) throws IOException {
  3. BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
  4. String str = br.readLine();
  5. System.out.println(str);
  6. br.close();
  7. BufferedReader bb = new BufferedReader(new InputStreamReader(System.in));
  8. str = bb.readLine();
  9. System.out.println(str);
  10. bb.close();
  11. }
  12. }

使用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进行搭配使用,代码如下:

  1. java复制代码import java.io.BufferedReader;
  2. import java.io.IOException;
  3. import java.io.InputStream;
  4. import java.io.InputStreamReader;
  5. class NotCloseISR extends InputStreamReader{
  6. public NotCloseISR(InputStream in){
  7. super(in);
  8. }
  9. public void close() throws IOException {
  10. //do nothing
  11. }
  12. }
  13. public class Test {
  14. public static void main(String[] args) throws IOException {
  15. testInput(); // 两次调用该函数,但每次调用后仅关闭BufferedReader的流而不关闭System.in流
  16. testInput();
  17. }
  18. public static void testInput() throws IOException {
  19. //对于传入的System.in做包装
  20. BufferedReader reader = new BufferedReader(new NotCloseISR(System.in)); //
  21. System.out.println("请输入一行字符:");
  22. String str = reader.readLine();
  23. System.out.println("输入的字符:" + str);
  24. reader.close(); //
  25. }
  26. }

上述代码便不会关闭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

发表评论

表情:
评论列表 (有 0 条评论,5人围观)

还没有评论,来说两句吧...

相关阅读

    相关 CentOS拷贝启动问题解决

    在需要多台虚机配置完成分布式环境的时候,需要安装多台虚机,在已经安装好一台虚机的基础上,通过复制的方式,连续创建多台虚机是一个较为方便的方式。但是,复制的方式会将原始虚机的网卡