Android 3.x + Java ZipFile Class - Невозможно прочитать ZipEntries из больших файлов
Если я открою большой zip файл (250 МБ) через класс ZipFile и попробую прочитать записи. Это отлично работает на 2.x в эмуляторе и реальном оборудовании. Если я использую точный код на моем планшете (Asus Transformer 4.0.3) или эмулятор (3.2), я не могу прочитать никаких записей. Функция size() класса ZipFile всегда возвращает ноль, а ZipFile не возвращает записи zip. Даже приложение zip, поставляемое с ROM на моем планшете, не может читать записи. Почтовый файл не поврежден. Я проверил его.
Код для чтения из ZipFile отлично работает на всех версиях с меньшими zip файлами. Что изменилось между 2.x и 3.x/4.x??
Мой тестовый файл - это C64Music.zip из коллекции HighVoltage Sid. Он содержит более 40 000 файлов и составляет около 250 МБ.
Я понятия не имею, на что посмотреть.
Ответы
Ответ 1
Это известная проблема с реализацией приложения для Android:
http://code.google.com/p/android/issues/detail?id=23207
В основном файлы zip поддерживают только до 65 тыс. записей. Существует расширенная версия zip файла формата Zip64, который поддерживает большее количество записей. К сожалению, ZipFile на Android не может читать Zip64. Вероятно, вы обнаружите, что файл C64Music.zip находится в формате Zip64
Обходной задачей является использование библиотеки Apache Commons Compress вместо собственной реализации. Их версия ZipFile поддерживает Zip64: http://commons.apache.org/compress/apidocs/org/apache/commons/compress/archivers/zip/ZipFile.html
Ответ 2
public class Compress {
private static final int BUFFER = 2048;
private String[] _files;
private String _zipFile;
public Compress(String[] files, String zipFile) {
_files = files;
_zipFile = zipFile;
}
public void zip() {
try {
BufferedInputStream origin = null;
FileOutputStream dest = new FileOutputStream(_zipFile);
ZipOutputStream out = new ZipOutputStream(new BufferedOutputStream(dest));
byte data[] = new byte[BUFFER];
for(int i=0; i < _files.length; i++) {
Log.v("Compress", "Adding: " + _files[i]);
FileInputStream fi = new FileInputStream(_files[i]);
origin = new BufferedInputStream(fi, BUFFER);
ZipEntry entry = new ZipEntry(_files[i].substring(_files[i].lastIndexOf("/") + 1));
out.putNextEntry(entry);
int count;
while ((count = origin.read(data, 0, BUFFER)) != -1) {
out.write(data, 0, count);
}
origin.close();
}
out.close();
} catch(Exception e) {
e.printStackTrace();
}
}
}
Call Compress like given below where you want to zip a file :----
String zipFilePath = "fileName.zip";
File zipFile = new File(zipFilePath);
String[] files = new String[] {"/sdcard/fileName"};
if(!zipFile.exists()){
new Compress(files, zipFilePath).zip();
}
Ответ 3
В 3.x/4.x было внесено много изменений, чтобы предотвратить злоупотребление потоком пользовательского интерфейса. Таким образом, возможно, что ваше приложение рушится, потому что вы не выгружаете дорогостоящую операцию ввода-вывода диска в отдельный Thread
.