java

关注公众号 jb51net

关闭
首页 > 软件编程 > java > System.in关闭后无法使用流

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

作者:陈宇超

这篇文章主要给大家介绍如何解决System.in关闭后无法再继续使用流的问题,文中有详细的解决方法和代码示例,具有一定的参考价值,需要的朋友可以参考下

解决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

到此这篇关于一文解决System.in关闭后无法再继续使用流的问题的文章就介绍到这了,更多相关System.in关闭后无法使用流内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

您可能感兴趣的文章:
阅读全文