当我像这样将条目写入 zip 文件时:
ZipEntry ze = zin.getNextEntry();
while (ze != null) {
InputStream is = zf.getInputStream(ze);
zos.putNextEntry(ze);
int len;
while ((len = is.read(buffer)) >= 0) {
zos.write(buffer, 0, len);
}
zos.closeEntry();
ze = zin.getNextEntry();
}
我在第二个 while 循环中得到以下异常:
java.util.zip.ZipException: invalid code lengths set
at java.util.zip.InflaterInputStream.read(InflaterInputStream.java:164)
at java.io.FilterInputStream.read(FilterInputStream.java:107)
有人知道为什么会抛出这个异常,它是什么意思?
附言我应该提一下,我正在 JBoss 7.1.1 的监听器上运行它,以便从不同的文件夹压缩各种日志文件。每个文件夹都有一个线程。使用多线程的事实会导致这个问题吗?
最佳答案
您正在将新 zip 文件的 ZipEntry
设置为您从原始文件中获得的相同实例。这意味着所有值都必须匹配,但如果写入条目的压缩大小与源文件的压缩大小不匹配,则会失败。原始压缩代码和您现在使用的压缩代码之间最细微的差别都会产生不同的压缩大小。要使其运行,您必须为输出创建 ZipEntry
的副本并重置其压缩大小。
顺便说一句,您在代码中使用了 zin.getNextEntry()
和 zf.getInputStream(ze)
,这意味着您正在使用 ZipFile
和一个 ZipInputStream
同时。如果它们引用同一个文件,那是一种资源浪费,如果它们引用不同的文件,您可能会遇到更多问题。
决定使用 ZipFile
ZipFile zf = …
ZipOutputStream zos = …
byte[] buffer = …
for(Enumeration<? extends ZipEntry> e=zf.entries(); e.hasMoreElements();) {
ZipEntry ze = e.nextElement();
InputStream is = zf.getInputStream(ze);
ZipEntry zeOut=new ZipEntry(ze);
zeOut.setCompressedSize(-1);
zos.putNextEntry(zeOut);
int len;
while ((len = is.read(buffer)) >= 0) {
zos.write(buffer, 0, len);
}
zos.closeEntry();
}
zos.close();
zf.close();
或ZipInputStream
ZipInputStream zin…
ZipOutputStream zos=…
byte[] buffer = …
ZipEntry ze = zin.getNextEntry();
while (ze != null) {
ZipEntry zeOut=new ZipEntry(ze);
zeOut.setCompressedSize(-1);
zos.putNextEntry(zeOut);
int len;
while ((len = zin.read(buffer)) >= 0) {
zos.write(buffer, 0, len);
}
zos.closeEntry();
ze = zin.getNextEntry();
}
zos.close();
zin.close();
关于java: Zip Exception 无效代码长度?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19631409/