4

动机

我有一个SomeObject.java文件:

class SomeObject {
   String name;
}

编译它会创建一个包含字节码的SomeObject.class文件。

0xCAFEBABE...

如果我们在 JVM 上使用 SomeObject,它会被当前的类加载器加载并且一切正常。

现在让我们假设我想要一些动态代码生成。我可以编写我的自定义注释

@Target(ElementType.TYPE)
public @interface Data {
   ...
}

并将其作为修饰符添加到类声明中:

@Data
class SomeObject {
   String name;
}

我也可以在运行时保留它@Retention(RetentionPolicy.RUNTIME)

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
public @interface Data {
   ...
}

问题

用于字节码注入的注解在哪里?当使用适当的运行时保留注释加载类时,类加载器是否注入字节码,如下图所示:

source -(compile)-> bytecode -(classloader bytecode injection)-> injected bytecode -(classloading)-> JVM loaded bytecode   
4

1 回答 1

6

是的,可以让您的自定义类加载器加载一个类,并通过JavassistASM等字节码操作工具执行修改,而不是将类文件中的字节码加载到内存中,而是将修改后的字节码加载到内存中。尽管有更简单(我认为更好)的方法来做到这一点。

注释处理器工具 (APT)

从 Java 6 开始,您拥有APT,它允许您连接到编译过程(通过-processor javac 中的参数)。使用 APT,您可以访问代码的 AST(抽象语法树),并且可以在使用javax.lang.model进行编译时直接进行修改。这意味着将生成您的类文件并进行所需的修改。

在这种情况下,链将类似于:

source -(compile and performs modifications at model level)-> bytecode already modified - regular class loader -> loads class into memory

编译后处理

另一种可以使用的方法是在编译后作为编译后过程执行字节码注入。在这种情况下,您使用字节码修改工具(同样是 javassist、asm 等),它可以在找到所需的注解时执行您需要的修改,并使用注入的字节码生成一个新的类文件。

在这种情况下,您的链条将是:

source -compile -> bytecode -post-compile-> modified bytecode - regular class loader -> loads class into memory

运行时修改

最后我们到达运行时字节码修改。尽管您的想法是可能的,但在我看来,我会保留类加载器的魔力并使用 Javassist 之类的工具,它还允许您拥有可以修改和重新加载的动态代理

在 javassist 特定情况下,链将是

source -compile -> bytecode -post-compile-> modified bytecode - regular class loader -> loaded into memory - javassist proxy -> modified class - javassist hot swapper -> re-modified class

代理并不完美(当然没有什么是完美的)。您将受到性能影响,并且您将无法修改类的公共接口(旁注:APT 和后编译过程都允许您修改类公共接口)。我可以继续讨论这个问题,但我认为这已经足够让你深思了。如果您需要更多信息,请随时发表评论。

于 2013-03-07T13:22:16.687 回答