java - 无效条目压缩大小

标签 java exception jar java-bytecode-asm

我正在使用称为 ASM 的字节码库来更改类文件,然后我想将每个类文件写回一个 jar 文件,而不是一个充满类文件的文件夹。我通过运行这段代码来做到这一点:

当 ZipException 不是预期大小而抛出时,我的问题就出现了,即

java.util.zip.ZipException: invalid entry compressed size (expected 695 but got 693 bytes)
    at java.util.zip.ZipOutputStream.closeEntry(Unknown Source)
    at org.steinburg.client.accessor.Accessor.accessJar(Accessor.java:64)
    at org.steinburg.client.accessor.Accessor.<init>(Accessor.java:41)
    at Loader.main(Loader.java:5)


package org.steinburg.client.accessor;

import java.io.DataInputStream;
import java.io.File;
import java.io.FileOutputStream;
import java.io.InputStream;
import java.util.Enumeration;
import java.util.jar.JarEntry;
import java.util.jar.JarFile;
import java.util.jar.JarOutputStream;

import org.objectweb.asm.ClassReader;
import org.objectweb.asm.ClassWriter;

public class Accessor {

private String input;
private File inFile;
private JarEntry jarEntry;
private JarFile jarFile;
private Enumeration<JarEntry> entries;
private InputStream is;

private String out;
private File outFile;
private FileOutputStream fos;
private JarOutputStream jos;

private byte[] bytes;

public Accessor(){
    try{
        input = new String("Input Jar.jar");
        inFile = new File(input);
        jarFile = new JarFile(inFile);
        entries = jarFile.entries();
        out = new String("Output Jar.jar");
        outFile = new File(out);
        fos = new FileOutputStream(outFile);
        jos = new JarOutputStream(fos);
        accessJar();
    } catch (Exception e) {
        e.printStackTrace();
    }
}

protected final void accessJar(){
    try {
        while (entries.hasMoreElements()){
            jarEntry = entries.nextElement();
            is = jarFile.getInputStream(jarEntry);
            if (jarEntry.getName().endsWith(".class")){
                ClassReader cr = new ClassReader(is);
                ClassWriter cw = new ClassWriter(cr, 0);
                FieldAdapter fa = new FieldAdapter(cw);
                cr.accept(fa, 0);
                bytes = cw.toByteArray();
            } else {
                bytes = readBytes(is);
            }
            JarEntry je = new JarEntry(jarEntry);
            jos.putNextEntry(je);
            jos.write(bytes);
            jos.closeEntry();
        }
        jos.close();
    } catch (Exception e) {
        e.printStackTrace();
    }
}

protected byte[] readBytes(InputStream inputStream){
    try{
        DataInputStream reader = new DataInputStream(inputStream);
        byte[] toReturn = new byte[reader.available()];
        reader.readFully(toReturn);
        reader.close();
        return toReturn;
    } catch (Exception e){
        e.printStackTrace();
        return null;
    }
}

}

ClassReader、ClassWriter(库的每个部分)和 FieldAdapter(我自己制作的)只是提供代码,为了获取整个类的字节,我使用 cw.toByteArray()。对于字节码操作本身没有问题,它只是通过 JarOutputStream 写入一个新的 jar 文件。任何人都知道如何解决这个问题?

最佳答案

有问题的行是这样的:

JarEntry je = new JarEntry(jarEntry);

因为它还会复制压缩后的大小。

除非您关心保留名称以外的其他字段,否则您应该使用 this constructor并仅提供文件名,如下所示:

JarEntry je = new JarEntry(jarEntry.getName ());

然后将自动计算压缩后的大小。

关于java - 无效条目压缩大小,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/13435683/

相关文章:

c# - CollectionView.DeferRefresh() 抛出异常

c++ - C++中的异常是否仅用于提高可读性?

Java JAR 库压缩

java - UDP 中的服务器-客户端

java - 使用java api查找elasticsearch嵌套聚合的文档计数

c++ - Windows/MinGW下相当于 "SIGINT"(posix)信号,用于捕获 "CTRL+C"

java - 我应该将文件放在哪里,以便在运行 jar 时能够访问它们?

java - 可运行jar文件创建-jar包中包含的资源文件夹

java - 如何将一组 char[] 添加到 List<String> 中?

java - 将 bigint 转换为 long