我正在使用这段复制大文件。 Android正好在32个缓冲区加载时因“内存不足”而崩溃。它就像dos.write将数据放入一个大缓冲区,而不是将其假脱机到i / o设备。没有例外。
bufferSize = 512 * 1024。 bis是BufferedInputStream。 byteArray是一个ByteArrayBuffer,
try {
FileOutputStream fos = new FileOutputStream(file);
dos = new DataOutputStream(fos);
int current = 0;
while((current = bis.read()) != -1){
byteArray.append((byte)current);
if (byteArray.isFull()){
byte[] b = byteArray.toByteArray();
dos.write(b, 0, bufferSize);
byteArray.clear();
}
}
int count = byteArray.length();
byte[] b = byteArray.toByteArray();
dos.write(b, 0, count);
dos.flush();
dos.close();
bis.close();
}
catch (Exception e) {
RunTimeError("Exception: " + e);
return false;
}
答案 0 :(得分:7)
我的猜测是byteArray.isFull()
因某种原因总是返回false
。然后当你加载了16MB的数据时,你就会内存不足。我不打扰ByteArrayBuffer
。 (就此而言,512KB对于这种操作来说太大了。你应该尝试匹配文件I / O块大小。它可能因设备而异,但4K-8K可能很接近。)你也不要需要将fos
包裹在DataOutputStream
中;你只是在写字节。另一方面,BufferedOutputStream
可能有用。如果bis
未被缓冲,则将其包裹在BufferedInputStream
中也会有所帮助。
我会像这样重写你的代码:
BufferedOutputStream bos = null;
try {
bos = new BufferedOutputStream(new FileOutputStream(file), 8192);
byte[] buffer = new byte[1024];
int len = 0;
while((len = bis.read(buffer)) != -1) {
bos.write(buffer, 0, len);
}
} catch (Exception e) {
RunTimeError("Exception: " + e);
return false;
} finally {
try { bis.close(); } catch (Exception ignored) { }
try { bos.close(); } catch (Exception ignored) { }
}