java.util.zip.ZipException: слишком много записей в ZIP-файле

Я пытаюсь написать класс Java для извлечения большого zip-файла, содержащего ~ 74000 файлов XML. Я получаю следующее исключение при попытке распаковать его с помощью библиотеки java zip:

java.util.zip.ZipException: слишком много записей в ZIP-файле

К сожалению, из-за требований проекта я не могу разбить zip-архив до того, как он попадет ко мне, и процесс распаковки должен быть автоматизирован (никаких ручных действий). Есть ли способ обойти это ограничение с помощью java.util.zip или какой-либо сторонней библиотеки Java zip?

Спасибо.


person Andrew    schedule 27.01.2009    source источник


Ответы (4)


Вероятно, это должно сделать использование ZipInputStream вместо ZipFile.

person Tom Hawtin - tackline    schedule 27.01.2009

Использование apache IOUtils:

FileInputStream fin = new FileInputStream(zip);
ZipInputStream zin = new ZipInputStream(fin);
ZipEntry ze = null;

while ((ze = zin.getNextEntry()) != null) {
    FileOutputStream fout = new FileOutputStream(new File(
                    outputDirectory, ze.getName()));

    IOUtils.copy(zin, fout);

    IOUtils.closeQuietly(fout);
    zin.closeEntry();
}

IOUtils.closeQuietly(zin);
person Andrew    schedule 29.06.2009

Стандарт Zip поддерживает максимум 65536 записей в файле. Если библиотека Java не поддерживает расширения ZIP64, она не будет работать должным образом, если вы попытаетесь прочитать или записать архив с 74 000 записей.

person Cheeso    schedule 10.02.2009

Я переработал метод, чтобы упростить работу со структурой каталогов и заархивировать сразу целую кучу целей. Обычные файлы будут добавлены в корень zip-файла, если вы передадите каталог, основная структура будет сохранена.

def zip (String zipFile, String [] filesToZip){ 
    def result = new ZipOutputStream(new FileOutputStream(zipFile))
    result.withStream { zipOutStream ->
        filesToZip.each {fileToZip ->
            ftz = new File(fileToZip)
            if(ftz.isDirectory()){
                pathlength = new File(ftz.absolutePath).parentFile.absolutePath.size()
                ftz.eachFileRecurse {f ->               
                    if(!f.isDirectory()) writeZipEntry(f, zipOutStream, f.absolutePath[pathlength..-1]) 
                }
            }               
            else writeZipEntry(ftz, zipOutStream, '')
        }
    }
}

def writeZipEntry(File plainFile, ZipOutputStream zipOutStream, String path) {
    zipOutStream.putNextEntry(new ZipEntry(path+plainFile.name))
    new FileInputStream(plainFile).withStream { inStream ->
        def buffer = new byte[1024]
        def count
        while((count = inStream.read(buffer, 0, 1024)) != -1) 
            zipOutStream.write(buffer)                  
    }
    zipOutStream.closeEntry()
}
person Community    schedule 03.08.2009