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

2022-08-31 06:59:50

运行 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)

这是自定义活动的第 215 行.java:

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

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

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

  • 检查Android项目属性,确保正确添加了appcompat库。
  • 检查 Java 构建路径顺序和导出项目属性,确保选中 Android 依赖项和 Android 私有库。
  • 确认该类包含在库中(android.support.v7.internal.view.menu.Menu.Menubuilder)。
  • 确认 R.java位于 android.support.v7.appcompat 的 gen 目录中。
  • 确认 AppCompat 主题是否包含在清单.xml活动中。
  • 清理并重新生成项目。

尽管采取了这些步骤,尽管它适用于所有其他设备和Android版本,但崩溃报告仍然会出现。


答案 1

编辑:

对我有用的解决方案是(使用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.** {*;}

功劳归谷歌组#138

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

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);
}

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


答案 2

正如谷歌群组的#150所说

因为小心 -keep 类 !android.support.v7.internal.view.menu.**.其中有许多类是从 appcompat 的资源中引用的。

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

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

推荐