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

2023年 7月 14日 52.0k 0

解决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.outSystem.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()就会关闭BufferedReaderInputStreamReaderSystem.in三个流,此时若再使用BufferedReaderSystem.in进行包装则会报java.io.IOException: Stream closed,这是因为调用reader.close()后又级联调用了System.inclose方法,因此System.in已经被关闭了,无法再次使用。

解决办法就是自己构造一个类来连接BufferedReaderSystem.in两个流,这个类需要是InputStreamReader的子类(因为要连接BufferedReaderSystem.in,此外在这个类中要重写(覆盖)close方法使其不调用System.inclose方法(将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社区

相关文章

JavaScript2024新功能:Object.groupBy、正则表达式v标志
PHP trim 函数对多字节字符的使用和限制
新函数 json_validate() 、randomizer 类扩展…20 个PHP 8.3 新特性全面解析
使用HTMX为WordPress增效:如何在不使用复杂框架的情况下增强平台功能
为React 19做准备:WordPress 6.6用户指南
如何删除WordPress中的所有评论

发布评论