从检测方法调用自己的类时,Java NoClassDefFoundError

我正在研究一套简单的Java代理,以帮助我(并希望其他人)对Java应用程序进行故障排除。 我想创建的一个代理工具JComponent.getToolTipText()方法通过将鼠标光标hover在其上来快速识别任何GUI类。

你可以在这里找到变压器的代码和项目的其余部分:

http://sfn.cvs.sourceforge.net/viewvc/sfn/core/src/main/java/org/leplus/sfn/transformer/JComponentTransformer.java?view=markup

我使用附件代理启动我的测试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类。

这是应用程序在我启动时打印的内容,我把鼠标放在它上面:

加载代理:JComponent
仪表准备好了!
线程“AWT-EventQueue-0”中的exceptionjava.lang.NoClassDefFoundError:org / leplus / sfn / tracer / ComponentTracer
 在javax.swing.JComponent.getToolTipText(JComponent.java)
 在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: -javaagent:... 

尝试使用-DDEBUG运行,因为它可能会向您显示更多信息。

另外,我在这里看到目标目录。 http://sfn.cvs.sourceforge.net/viewvc/sfn/core/target/它包含一个classes文件夹,但是没有jars文件夹? 检查以确保jar路径相对于项目根目录。