作为我的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函数?
答案 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。然后在该方法的开头,检查该布尔值是否为真。如果是,则从方法返回,这样它就不会再次执行。