我正在开发一套简单的 Java 代理,以帮助我(以及希望其他人)对 Java 应用程序进行故障排除。我想创建工具 JComponent.getToolTipText() 方法的代理之一,只需将鼠标光标悬停在它上面即可快速识别任何 GUI 类。
你可以在这里找到我的转换器的代码和项目的其余部分:
我启动我的测试 GUI 并附加代理如下:
$ java -javaagent:target/jars/sfn-0.1-agent.jar=JComponent -cp lib/jars/bcel-5.2.jar:target/jars/sfn-0.1-test.jar:target/jars/sfn-0.1-agent.jar org.leplus.sfn.test.Main
sfn-0.1-agent.jar 包含 org.leplus.sfn.transformer.JComponentTransformer 类。 sfn-0.1-test.jar 包含 org.leplus.sfn.test.Main 类。
这是应用程序在我启动并将鼠标放在它上面时打印的内容:
Loading agent: JComponent Instrumentation ready! Exception in thread "AWT-EventQueue-0" java.lang.NoClassDefFoundError: org/leplus/sfn/tracer/ComponentTracer at javax.swing.JComponent.getToolTipText(JComponent.java) at javax.swing.ToolTipManager$insideTimerAction.actionPerformed(ToolTipManager.java:662) ...
令我惊讶的是,如果我更改我的转换器以从 JRE 调用任何类,它就可以工作。但是当我调用我自己的类 org.leplus.sfn.tracer.ComponentTracer 时它不起作用。我的第一个猜测是类路径问题,但 ComponentTracer 既在类路径中又在代理的 jar 中。所以我迷路了。
如果你们中有人看到我遗漏了什么。
干杯,
汤姆
最佳答案
这是一个类加载器问题。您正在检测由引导类加载器管理的类 (javax.swing.JComponent),并让它引用由系统类加载器管理的类 (org.leplus.sfn.tracer.ComponentTracer)。
如果将 ComponentTracer 类放在引导类加载器中,问题应该会消失。
java -Xbootclasspath/p:<path/to/jar/containing/ComponentTracer> -javaagent:...
关于从检测方法调用自己的类时出现 Java NoClassDefFoundError,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/4575761/