扩展读者,如何返回我的"阅读"?

时间:2015-04-30 21:19:34

标签: java

作为我的Java课程的一部分,我写了一个" zip"作家和读者 - 就像霍夫曼算法一样。

我的类扩展了Reader,并有一个对象Reader r。 在我的主要方法中,我有以下几行:

input = new BufferedReader(new HuffmanReader(new FileReader("output.hff")));
String str = input.readLine();

当然,它应该在解压缩之后返回我写入文件的解压缩字符串。但它返回文件的第一行!

我的阅读功能:

public int read(char[] cbuf, int off, int len) throws IOException {
    //...
    r.read(buffer,0,8192)
    //does the decompress process
    String fnlStr = ... //The final result
    cbuf = fnlStr.toCharArray();
    //close streams
    return cbuf.length;
}

我的调试窗口显示:

HuffmanReader.read(char[], int, int) line: 23   
BufferedReader.fill() line: not available   
BufferedReader.readLine(boolean) line: not available    
BufferedReader.readLine() line: not available   
Run.main(String[]) line: 23

它调用我的读取功能两次。如何阻止bufferReader再次调用read函数?

2 个答案:

答案 0 :(得分:2)

您不会像通常那样返回从方法中读取的数据。相反,当调用read时,调用者会为您提供数组cbuf,它基本上是一块内存的地址,并告诉您编写len char s进入它。

执行cbuf = fnlStr.toCharArray()时,您只是用另一个地址替换该地址的本地副本,但实际上并没有更改您应该写入的内存。您需要迭代在for循环中给出的数组并写入它,或者如果您构造了另一个包含结果的缓冲区,则使用System.arraycopy

例如,以下read方法将始终为"Test\n"

public int read(char[] cbuf, int off, int len) throws IOException {
    char[] result = "Test\n".toCharArray();
    int numRead = Math.min(len, result.length);
    System.arraycopy(result, 0, cbuf, off, numRead);
    return numRead;
}

用解压缩的字符串替换"Test\n"文字可以帮助您入门。当然,您仍然需要管理已经消耗了多少来源。

至于BufferedReader两次调用read:你不应该关心它的调用频率。只需从底层源获取数据,将其写入cbuf并返回您编写的char个数。如果没有任何内容可供阅读,请返回-1以表示信号流的结尾(在这种情况下,BufferedReader将停止呼叫read)。

顺便说一句,Reader用来读取字符流,而InputStream is for binary data(它基本上是相同的,只有byte[]而不是char[]而没有使用字符集)。由于压缩文件是二进制文件,因此您可能需要将FileReader切换为FileInputStream

如果由于某种原因,您编码的字符集与您解码时的字符集不同,我可以想象出奇怪的错误。如果UTF-16中的一个16位代码单元需要UTF-8中的3个8位代码单元,那么你可能会使用比你想象的更多的空间。

答案 1 :(得分:1)

您只阅读第一行。将第一部分更改为:

input = new BufferedReader(new HuffmanReader(new FileReader("output.hff")));
Arraylist<String> list = new ArrayList<String>();
String line;

while ((line = reader.readLine()) != null) {
    list.add(line);
}

而且,为了修复你的方法被调用两次,做一个布尔值并在你完成方法之后将它设置为true。然后在该方法的开头,检查该布尔值是否为真。如果是,则从方法返回,这样它就不会再次执行。