底层流返回零字节

时间:2012-06-06 15:57:58

标签: java io inputstream filestream

我目前正在用Java编写自己的Brainfuck解释器,因为我希望它能够从文件中读取代码,我写了BFInputStream来过滤掉不必要的符号。它看起来像这样:

import java.io.FilterInputStream;
import java.io.InputStream;
import java.io.IOException;

public class BFInputStream extends FilterInputStream {
    public BFInputStream(InputStream in) {
        super(in);
    }
    public int read() throws IOException {
        while(true) {
            int i = super.read();
            // keep LF for line number checking.
            if(i == -1 || i == 10 ||( i >= 43 && i <= 46) || i == 60 || i == 62 || i == 91 || i == 93)
                return i;
        }
    }
    public int read(byte[] b, int off, int len) throws IOException {
        if(off < 0 || len < 0 || len > b.length - off) throw new IndexOutOfBoundsException();
        for(int i=0; i<len; i++) {
            int j = read();
            if(j < 1) return i;
            b[off+i] = (byte)j;
        }
        return len;
    }
}

我的口译员使用LineNumberReader&lt; - InputStreamReader&lt; - BFInputStream&lt; - FileInputStream来阅读该文件。但每当它到达文件的末尾时它就抛出:

  

java.io.IOException:底层输入流返回零字节

  at sun.nio.cs.StreamDecoder.readBytes(StreamDecoder.java:268)
  at sun.nio.cs.StreamDecoder.implRead(StreamDecoder.java:306)
  at sun.nio.cs.StreamDecoder.read(StreamDecoder.java:158)
  at java.io.InputStreamReader.read(InputStreamReader.java:167)
  at java.io.BufferedReader.fill(BufferedReader.java:136)
  at java.io.BufferedReader.read(BufferedReader.java:157)
  at java.io.LineNumberReader.read(LineNumberReader.java:108)
  at Interpreter.run(Interpreter.java:101)
  at Interpreter.main(Interpreter.java:180)

Interpreter.java:101包含对BFInputStream.read()的调用。

我不确定我是否正确理解了异常。我认为流根本没有返回任何字节(因为超时)。我想如果有一大块非BF字符,那么这可能是可能的但是在文件的末尾? FileInputStream因此FilterInputStreamBFInputStream应返回-1。

我的代码出了什么问题?

2 个答案:

答案 0 :(得分:2)

if (j < 1)应为if (j < 0),因为EOF由-1表示。此外,您的方法永远不会在EOF返回-1。

正确的实现如下所示:

public int read(byte[] b, int off, int len) throws IOException {
    if(off < 0 || len <= 0 || len > b.length - off) throw new IndexOutOfBoundsException();         
    int i = 0;
    while (i < len) {
        int j = read();
        if (j < 0) break; // Stop reading at EOF
        b[off + i] = (byte) j;
        i++;
    }
    if (i == 0) return -1; // If we get EOF with no data, return it to the caller
    else return i;
}

答案 1 :(得分:1)

之前使用File,FileReader / Writer和BufferReader / Writer做了同样的事情。 我给你的代码片段我用作指南..试试吧

从文件中读取

File f = new File("my.txt");
FileReader fr = new FileReader(f);
BufferedReader br  = new BufferedReader(fr);

String s = null;

while ((br=readLine())!=null) {

// Do whatever u want to do with the content of the file.

}

br.close();

写入文件:

Boolean isDone = true;
Scanner scan = new Scanner(System.in);
File f = new File("my.txt");
FileWriter fr = new FileWriter(f);
BufferedWriter br  = new BufferedWriter(fr);

while (b) {

   if (!b) {

 br.write(new Scanner(System.in).nextLine());

 }


}