目前我正在尝试修改驻留在JVM已经加载的类中的方法体。我知道JVM实际上不允许更改已经加载的类的定义。但是我的研究使我得到了像JRebel或Java Instrumentation API这样的实现,两者都使用了基于代理的方法。我知道如何在代表Javassist加载类之前做到这一点。但考虑到例如在应用程序启动时加载类定义的EJB环境中的JRebel,不应该在加载JVM的类上进行字节码修改吗?
好吧,你了解到Instrumentation API存在并且它提供redefinition of classes作为操作。那么现在是时候重新考虑“JVM实际上不允许更改已经加载的类的定义”的初始前提。
你应该注意到这一点
因此,Instrumentation仅用于调试和分析等。
但是其他框架,比如EJB容器,在生产代码中提供类重新加载,通常需要创建新的ClassLoader
s,它们创建不同版本的类,然后完全独立于旧版本。
在Java运行时环境中,类的标识由一对<ClassLoader, Qualified Name>
组成,而不仅仅是一个限定名称...
我不知道您可以使用instrumentation API来重新定义类(请参阅@ Holger的答案)。但是,正如他所指出的那样,这种方法存在一些重大限制。此外,javadoc说:
“此方法适用于仪器,如类规范中所述。”
使用它来实质上改变类的语义是......从Java类型系统的角度来看各种各样的错误。