我正在尝试从文件“A”中读取字节。我正在使用RandomAccessFile来寻找一个位置,然后我需要读取“y”字节。
我有一个4096字节的缓冲区,如果“y”不是4096的倍数,我读取的字节比我应该多。
如果我将缓冲区设置为1个字节,我可以毫无问题地进行读写(但当然,它太慢了)。
我目前的代码是:
public void extractFile(int pos) {
try {
RandomAccessFile raf = new RandomAccessFile(this.archive, "r");
/* Trying to read */
raf.seek(arquivos.get(pos).getPosicaoInicio());
ByteArrayOutputStream byteOutput = new ByteArrayOutputStream();
byte[] buf = new byte[1]; // With 1 I can read, because every "y" is multiple of 1
byte[] bytes;
while (byteOutput.size() < arquivos.get(pos).getTamanho()) {
byteOutput.write(buf, 0, raf.read(buf));
}
bytes = byteOutput.toByteArray();
byteOutput.close();
raf.close();
/* Writing */
File futuroArquivo = new File(arquivos.get(pos).getNome());
FileOutputStream fos = new FileOutputStream(futuroArquivo);
fos.write(bytes);
fos.flush();
fos.close();
} catch (IOException ex) {
}
}
PS:“arquivos.get(pos).getTamanho()”是我的“y”
PS 2:我无法读取整个文件,因为“y”字节后,还有其他的东西答案 0 :(得分:1)
缓冲区可以是零以上的任何大小,而ByteArrayOutputStream
实际上是浪费时间。和空间。您假设read()
也填充了缓冲区。写一个更好的方法是:
RandomAccessFile raf = new RandomAccessFile(this.archive, "r");
/* Trying to read */
raf.seek(arquivos.get(pos).getPosicaoInicio());
byte[] buf = new byte[8192]; // or more, whatever you like really
/* Writing */
File futuroArquivo = new File(arquivos.get(pos).getNome());
FileOutputStream fos = new FileOutputStream(futuroArquivo);
int count;
long rest = arquivos.get(pos).getTamanho();
while (rest > 0 && (count = raf.read(buf, 0, (int)Math.min(buf.length, rest))) > 0)
{
fos.write(buf, 0, count);
rest -= count;
}
fos.close();
raf.close();
我还会考虑在BufferedInputStream
周围使用FileInputStream
,而不是RandomAccessFile
。您并不是真正进行随机访问,只是初步搜索或跳过。
答案 1 :(得分:0)
根据EJP的评论,我已经设法解决了这个问题。我需要一些额外的过程,但至少我做到了。该解决方案需要一个辅助var(我称之为“rest”,只是改变了while循环,代码的另一部分是相同的):
long rest = arquivos.get(pos).getTamanho();
while (rest > 4096) {
rest -= 4096;
byteOutput.write(buf, 0, raf.read(buf));
}
byteOutput.write(buf, 0, raf.read(buf, 0, (int) rest));
PS:buf的大小需要为4096而不是1