运行时出现 NoClassDefFoundError 但类在 classes.dex 中,有啥好处?

Posted

技术标签:

【中文标题】运行时出现 NoClassDefFoundError 但类在 classes.dex 中,有啥好处?【英文标题】:NoClassDefFoundError at runtime but class is in classes.dex, what givies?运行时出现 NoClassDefFoundError 但类在 classes.dex 中,有什么好处? 【发布时间】:2015-04-25 12:58:05 【问题描述】:

我已经包含了一个从 Eclipse android 项目构建的 JAR 文件,我在我的 Android Studio 项目中引用它,如下所示:

  compile files('libs/libraryproject.jar')

这行得通,我可以在代码引用上自动完成。当我编译 APK 时,一切都很好。我安装并运行,然后收到 no class def 错误:

 java.lang.NoClassDefFoundError: com.android.canvas.CanvasContainer
            at com.app.MainActivity.onCreate(MainActivity.java:22)
            at android.app.Activity.performCreate(Activity.java:5231)
            at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1087)
            at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2159)
            at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2245)
            at android.app.ActivityThread.access$800(ActivityThread.java:135)
            at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1196)
            at android.os.Handler.dispatchMessage(Handler.java:102)
            at android.os.Looper.loop(Looper.java:136)
            at android.app.ActivityThread.main(ActivityThread.java:5017)
            at java.lang.reflect.Method.invokeNative(Native Method)
            at java.lang.reflect.Method.invoke(Method.java:515)
            at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:779)
            at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:595)
            at dalvik.system.NativeStart.main(Native Method)

但是,当我解压 APK 并使用 dexdump 查看 classes.dex 文件时,我看到上面的类存在那里。

那为什么我在运行时收到此类未找到错误?

更新

感谢 Chris 的建议,我注意到在前面的日志中,我的 CanvasContainer 类无法链接,因为它无法解析 interface 1990 'Lorg/cocos2dx/lib/Cocos2dxHelper$Cocos2dxHelperListener;'。很显然,我的 Cocos 库代码似乎没有作为我的 JAR 的一部分导出。

【问题讨论】:

CanvasContainer 的基类是什么?有没有可能它是来自比设备上的 API 更新的 API 的某个类? 一个在运行时无法满足依赖的类通常会在安装过程中被删除。如果你卸载应用,启动一些收集所有logcat的东西然后重新安装,你可能会通过搜索结果发现在安装过程中生成的日志中提到了埋藏的问题。 @ChrisStratton 很棒的一点,如果你愿意把它写成答案,我会接受。我现在看到由于没有解析接口 1990 'Lorg/cocos2dx/lib/Cocos2dxHelper$Cocos2dxHelperListener;'而无法链接我的类。 @Selvin 不,但你走在正确的轨道上。基类是一个 Cocos2d 库类,它没有与我的 JAR 文件一起导出。由于缺少 Cocos2d 接口,该异常不是在编译时发生,而是在运行时发生。 有时最好的办法是离开这个问题一晚上,在 SO 上听其他人的意见,然后在早上回来。如果不是 Chris 的建议,我早就注意到之前的链接失败了。 【参考方案1】:

一个在运行时无法满足的依赖类,通常会在安装、dex优化或类似的准备过程中被丢弃。

如果你卸载应用程序,启动一些收集所有logcat的东西然后重新安装它,你可以通过搜索结果发现在安装/准备过程中生成的日志中提到了埋藏的问题。

在此特定情况下,对问题的编辑表明缺少的类本身具有这种类型的未满足依赖关系。

【讨论】:

奇迹的奇迹 我多年后登录后发现您仅在三天前发布了此内容。感谢您很久以前的帮助!

以上是关于运行时出现 NoClassDefFoundError 但类在 classes.dex 中,有啥好处?的主要内容,如果未能解决你的问题,请参考以下文章

运行查询以创建表时出现错误 1064?

运行 Pig 脚本时出现异常

使用 htdocs xampp 运行命令时出现以下错误

为啥在运行 jupyter notebook 时出现错误?

尝试运行 jupyter notebook 时出现导入错误

运行 robolectric 时出现未知错误