在编译的类中没有反射的序列化

时间:2015-04-23 10:56:21

标签: java serialization code-generation bytecode-manipulation

由于客户端JVM的限制,我不能使用任何流行的序列化程序,因为不支持反射。我正在寻找一种工具,通过将编写器和读取器方法注入已编译的类来执行字节码操作以实现序列化。 我需要字节码操作java代码将它与我的构建过程代码绑定。

我一直这样做是通过生成代码并将其注入源代码,然后再编译以使用我的自定义序列化程序。我想避免这种方法,因为我不想以任何方式修改源文件。

我知道Kryo和其他XML和JSON序列化程序,但它们并不符合我的需求。

感谢。

3 个答案:

答案 0 :(得分:8)

试试javassist。它可能是您特定项目中最直接的字节码生成库。

那是因为你能够重用当前代码生成器的一部分,因为javassist能够解析一些简单形式的java代码。

例如,你可以这样做:

CtClass clazz = ...;
CtMethod m = CtNewMethod.make(
    "public void serialize(java.io.DataOutput out) {  out.writeInt(x); }",
     clazz);

如果您想深入了解字节码生成,可以试试asm。 Asm将允许你直接编写字节码,但它似乎对你的问题类型有点过分。

使用javassist实现

以下是使用javassist执行此操作的基本框架:

Path inputDir = Paths.get("target/classes");
Path outputDir = Paths.get("target/classes2");

ClassPool classPool = new ClassPool(true);
classPool.appendClassPath(inputDir.toString());

// get all class names from a certain directory
String[] classNames = Files.walk(inputDir)
        .filter(p -> (!Files.isDirectory(p)) && p.toString().endsWith(".class"))
        .map(p -> inputDir.relativize(p).toString())
        .map(s -> s.substring(0, s.length() - 6).replace(File.separatorChar, '.'))
        .toArray(size -> new String[size]);

for (String className : classNames) {
    CtClass clazz = classPool.get(className);
    // add further filtering to select the classes you want.

    // ex: "public void serializer(java.io.DataOutput out) { out.writeInt(x); } }"
    String serializerBody = generateSerializer(clazz);
    clazz.addMethod(CtNewMethod.make(serializerBody, clazz));

    // ex: "public void deserializer(java.io.DataInput in) { x = in.readInt(); } }";
    String deserializerBody = generateDeserializer(clazz);
    clazz.addMethod(CtNewMethod.make(deserializerBody, clazz));

    // save the modified class
    clazz.setModifiers(clazz.getModifiers() & ~Modifier.ABSTRACT);
    byte[] bytes = clazz.toBytecode();
    Path outFile = outputDir.resolve(className.replace('.', '/') + ".class");
    Files.createDirectories(outFile.getParent());
    Files.write(outFile, bytes);
}

依赖关系:org.javassist:javassist:3.19.0-GA

答案 1 :(得分:4)

您可以将我的库Byte Buddy用于此目的。 Byte Buddy是一个字节码操作库,它允许您轻松地向任何现有类添加方法。此外,它允许您inject redefined code into a jar file。这样,您可以要求Byte Buddy重新定义类以添加所需的方法。但是,请注意,向类添加方法可能会更改其隐式序列化uuid。

如果在应用程序启动之前不能加载类,Byte Buddy允许您重新定义类而不加载它们。为此,您可以使用Byte Buddys类型池。

答案 2 :(得分:1)

一些可能的替代方案:

  • 如果你可以使用scala,这个项目会编译时间序列化器生成: https://github.com/scala/pickling
  • 此stackoverflow问题指出使用aspectJ和注释处理的可能性: AspectJ / Generate methods using compile-time reflection
    顺便说一句,如果您的注释处理器声称要处理" *"那么您的课程不需要注释。所有注释。
  • Seren (SERialization ENhancer)声称​​"增强了您的课程,以便他们更快地序列化。" 然而,像许多其他工具一样,它会在加载时执行此操作。有可能让它适应构建时间仪表,或者向它的创建者提出建议。
  • leshy,也执行运行时检测。同样的事情:使其适应编译时间或向作者提出建议。