NoClassDefFoundError:android.support.v7.internal.view.menu.MenuBuilder

时间:2014-07-17 17:07:53

标签: java android eclipse android-appcompat

运行Android 4.2的三星设备上的Android appcompat v7库存在问题。我的开发人员控制台中的以下堆栈跟踪一直在崩溃:

java.lang.NoClassDefFoundError: android.support.v7.internal.view.menu.MenuBuilder
    at android.support.v7.widget.PopupMenu.<init>(PopupMenu.java:66)
    at com.[my-package-name].CustomActivity$5.onClick(CustomActivity.java:215)
    at android.view.View.performClick(View.java:4222)
    at android.view.View$PerformClick.run(View.java:17620)
    at android.os.Handler.handleCallback(Handler.java:800)
    at android.os.Handler.dispatchMessage(Handler.java:100)
    at android.os.Looper.loop(Looper.java:194)
    at android.app.ActivityThread.main(ActivityThread.java:5391)
    at java.lang.reflect.Method.invokeNative(Native Method)
    at java.lang.reflect.Method.invoke(Method.java:525)
    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:833)
    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:600)
    at dalvik.system.NativeStart.main(Native Method)

这是CustomActivity.java的第215行:

PopupMenu popup = new PopupMenu(CustomActivity.this, mImageViewMenu);

崩溃来自一系列设备,但始终是三星,而且始终是Android 4.2。

快速的网络搜索让我相信很多人都有同样的问题,我试图解决这个问题的一些步骤是:

  • 检查Android项目属性,确保正确添加appcompat库。
  • 检查Java Build Path Order和Export项目属性,确保选中Android Dependencies和Android Private Libraries。
  • 确认该类包含在库中(android.support.v7.internal.view.menu.MenuBuilder)。
  • 确认R.java位于android.support.v7.appcompat的基因目录中。
  • 确认AppCompat主题包含在Manifest.xml活动中。
  • 清理并重建项目。

尽管采取了这些步骤,尽管它在所有其他设备和Android版本上工作,但崩溃报告仍然存在。

10 个答案:

答案 0 :(得分:100)

编辑:

对我有用的解决方案是(使用Proguard)来替换它:

-keep class android.support.v4.** { *; } 
-keep interface android.support.v4.** { *; }

-keep class android.support.v7.** { *; }
-keep interface android.support.v7.** { *; }

用这个:

# Allow obfuscation of android.support.v7.internal.view.menu.**
# to avoid problem on Samsung 4.2.2 devices with appcompat v21
# see https://code.google.com/p/android/issues/detail?id=78377
-keep class !android.support.v7.internal.view.menu.**,android.support.** {*;}

积分转到google group, #138

旧答案(临时解决方法): 它发生在我在 ActionBar中使用微调器的项目中。我的解决方案是检查这些条件并更改应用程序流程:

public static boolean isSamsung_4_2_2() {
    String deviceMan = Build.MANUFACTURER;
    String deviceRel = Build.VERSION.RELEASE;
    return "samsung".equalsIgnoreCase(deviceMan) && deviceRel.startsWith("4.2.2");
}

然后在活动的onCreate方法中:

if (isSamsung_4_2_2()) {
    setContentView(R.layout.activity_main_no_toolbar);
} else {
    setContentView(R.layout.activity_main);
}

正如所指出的,这不是一个明确的解决方案,它只是一种允许用户访问有限功能同时找到更永久解决方案的方法。

答案 1 :(得分:26)

正如#150 from google groups所说

  

因为小心使用-keep class   !android.support.v7.internal.view.menu。**。有很多   那里的类​​是从appcompat的资源引用的。

更好的解决方案是添加以下行:

-keep class !android.support.v7.internal.view.menu.*MenuBuilder*, android.support.v7.** { *; }
-keep interface android.support.v7.** { *; }

答案 2 :(得分:23)

您遇到此问题的设备在哪? (三星/ HTC等)

如果是三星,

各种三星手机都包含在框架或类路径中的旧版Android支持库。如果您使用新材料支持库,则会在这些三星设备上看到此崩溃:

java.lang.NoClassDefFoundError: android.support.v7.internal.view.menu.MenuBuilder

要解决此问题,您必须重命名该类。最简单的方法是运行proguard。如果你不想混淆,那么这里只需要重写一个有问题的班级:

-keep class !android.support.v7.internal.view.menu.**,** {*;}

有一个跟踪此问题的问题,但由于它确实是一个三星的错误,它永远不会得到修复。在Google / AOSP方面修复它的唯一方法是重命名这些内部类。

https://code.google.com/p/android/issues/detail?id=78377

答案 3 :(得分:15)

此问题在AppCompat 23.1.1中返回,其中.internal包已从库jar中删除。

正如上面的评论中所建议的那样(在那里提出建议的人的信用),现在也必须改变proguard配置。

要获得上面建议的答案,请尝试将这些行添加到您的proguard文件中:

#FOR APPCOMPAT 23.1.1:
-keep class !android.support.v7.view.menu.*MenuBuilder*, android.support.v7.** { *; }
-keep interface android.support.v7.* { *; }

取代旧修正案:

#FOR OLDER APPCOMPAT VERSION:
-keep class !android.support.v7.internal.view.menu.*MenuBuilder, android.support.v7.** { ; }
-keep interface android.support.v7.* { *; }

答案 4 :(得分:12)

根据错误报告的最新帖子,这应该在新版本的支持库(24.0.0)上修复: https://code.google.com/p/android/issues/detail?id=78377#c374

有人even claimed修复了它。

此版本可用since last month,因此您应该更新它。

答案 5 :(得分:4)

是。三星已经知道this问题。 我建议您尝试使用GitHub中相同的Popup实现。这不是最好的方式,但会奏效。

答案 6 :(得分:4)

我遇到了在USB调试模式下找不到的这个MenuBuilder类的问题。我在 build.gradle 的发布和调试 buildTypes 块中简单地将 minifyEnabled 设置为 true 解决了这个问题。像这样:

buildTypes {

    debug {

        minifyEnabled true
    }

    release {
        minifyEnabled true
        proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
    }
}

我在调试类型中将 minifyEnabled 设置为true,以防止应用程序通过USB调试崩溃到现场手机。

答案 7 :(得分:0)

我使用eclipse项目提供的默认proguard属性启用了proguard,问题已经解决了。根据{{​​3}}的一些评论,有些人可能需要重新打包: -repackageclasses&#34; android.support.v7&#34;

答案 8 :(得分:0)

尝试通过Android Studio在我的Samsung Galaxy Tab 3平板电脑上运行“Hello World”应用时出现同样的错误。该应用程序似乎将启动,然后它会立即崩溃,并且该错误将显示在Android Studio的控制台中。我在平板电脑上进行了系统更新,现在我可以运行“Hello World”应用程序而且我不再收到错误了。我希望这有助于某人解决他们的问题。

注意:我在平板电脑上执行的系统更新未更新Android操作系统版本,因为它仍然说版本为4.2.2。

答案 9 :(得分:-4)

将项目的编译Sdk版本更改为&#34; API 18:(JellyBean)&#34;

默认设置为&#34; Lollipop

STEPS

  1. 右键单击您的项目,然后选择打开模块设置(或按F4)
  2. 在属性标签已编译的Sdk版本