按菜单按钮时应用程序崩溃

10
我正在尝试创建一个Android应用程序,但遇到了以下问题:
当我按菜单按钮时,该应用程序会在某些手机上崩溃。让我先给您一些详细信息。
  • 该错误仅在LG Optimus L3 II e430上出现,并且使用Android 4.1.2(迄今已在其他四部手机上测试)
  • 应用程序从闪屏开始,没有操作栏。此时,菜单按钮只是没有任何作用。
  • 通过简单的触摸,我们跳过闪屏,进入实现ActionBar活动和具有导航抽屉的主要活动。
  • 从这一点开始,在我尝试点击菜单按钮时,应用程序就会崩溃。

这是菜单的布局和onCreateOptionsMenu函数:

res/menu/main.xml

<menu xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto">
    <item android:id="@+id/action_settings"
        android:title="@string/action_settings"
        android:orderInCategory="100"
        app:showAsAction="never" />
</menu>

来自MainActivity.java的部分

    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        if (!mNavigationDrawerFragment.isDrawerOpen()) {
            // Only show items in the action bar relevant to this screen
            // if the drawer is not showing. Otherwise, let the drawer
            // decide what to show in the action bar.
            getMenuInflater().inflate(R.menu.main, menu);
            restoreActionBar();
            return true;
        }
        return super.onCreateOptionsMenu(menu);
    }

请注意,此代码是从Android Studio生成的。
到目前为止,我尝试过:
- 尝试查看具有问题的文件来自sdk源(API级别16和21),但它们与堆栈跟踪不相关(堆栈跟踪中显示的行指向一个不合理的位置)。 - 尝试安装XPosed修复Google PlayStore崩溃的菜单按钮错误。这里也没有找到解决方法。 - 在firefox的bugtracking系统中找到了一个类似的错误报告,因此我尝试安装Firefox并查看当我按Menu按钮时它是否会在我的手机上崩溃; firefox并没有崩溃。(Link to firefox's bug
以下是LogCat中的堆栈跟踪:
10-24 09:08:02.710    4712-4712/com.scaryboxstudios.unrealestateapp E/AndroidRuntime﹕ FATAL EXCEPTION: main
    java.lang.NullPointerException
            at com.android.internal.policy.impl.PhoneWindow.onKeyUpPanel(PhoneWindow.java:1004)
            at com.android.internal.policy.impl.PhoneWindow.onKeyUp(PhoneWindow.java:1712)
            at com.android.internal.policy.impl.PhoneWindow$DecorView.dispatchKeyEvent(PhoneWindow.java:2125)
            at android.view.ViewRootImpl.deliverKeyEventPostIme(ViewRootImpl.java:3611)
            at android.view.ViewRootImpl.handleImeFinishedEvent(ViewRootImpl.java:3581)
            at android.view.ViewRootImpl$ViewRootHandler.handleMessage(ViewRootImpl.java:2831)
            at android.os.Handler.dispatchMessage(Handler.java:99)
            at android.os.Looper.loop(Looper.java:137)
            at android.app.ActivityThread.main(ActivityThread.java:4929)
            at java.lang.reflect.Method.invokeNative(Native Method)
            at java.lang.reflect.Method.invoke(Method.java:511)
            at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:798)
            at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:565)
            at de.robv.android.xposed.XposedBridge.main(XposedBridge.java:132)
            at dalvik.system.NativeStart.main(Native Method)

2
发现针对支持库的问题已经被提交:https://code.google.com/p/android/issues/detail?id=78154 - TalkLittle
@TalkLittle 谢谢!我会关注的! - TheCrafter
1
LG L9在4.1.2上出现了同样的问题。 - Mikalai Daronin
3个回答

2
更新:在Appcompat-v7 22.0.0版本中,onKeyUp似乎不会对菜单键触发。原始错误似乎已经修复,所以我可能会删除子菜单的解决方法。不幸的是,我还没有在受影响的LG 4.1设备上验证修复程序。

最终我采用了一个解决方法,用户报告称这个方法已经解决了问题。

使用子菜单代替溢出菜单。缺点是现在每个设备都将在操作栏中看到溢出按钮,即使它们有一个菜单键。

以下技术来自https://dev59.com/M2ox5IYBdhLWcg3wTytx#18530179

  1. 将所有溢出选项菜单项转换为子菜单。

  2. 在您的活动中覆盖onKeyUp,让它调用Menu.performIdentifierAction(R.id.menu_overflow, 0);,并且不要为keyCode == KEYCODE_MENU调用super.onKeyUp


感谢您的回答,TalkLittle。您说的当然是正确的,但我不需要绕过它。事实上,重写onKeyUp是我想到的第一件事。主要问题是找到这个问题的根源并找到正确的解决方法,而不是避免它! - TheCrafter

0

同样,为了捕获菜单按钮,可以使用以下代码:

@Override
public boolean dispatchKeyEvent(KeyEvent event) {
    if (event.getKeyCode() == KeyEvent.KEYCODE_MENU) {


        // TODO - Your user code

        /*
        if (event.getAction() == KeyEvent.ACTION_DOWN
                && event.getRepeatCount() == 0) {

            // Tell the framework to start tracking this event.
            //getKeyDispatcherState().startTracking(event, this);
            return true;

        } else if (event.getAction() == KeyEvent.ACTION_UP) {

           // getKeyDispatcherState().handleUpEvent(event);
            if (event.isTracking() && !event.isCanceled()) {

                // DO BACK ACTION HERE
                return true;

            }
        }*/

        // if you don't want use system  listener 
        // return super.dispatchKeyEvent(event);
        return false;

    } else {
        return super.dispatchKeyEvent(event);
    }
}

适用于最新的AppCompat和SDK版本-22.0


没错,那个方法可以用,但是这只是个"权宜之计"。就像我在下面TalkLittle的回答中说的那样,我找到了许多临时解决方案。不过,我还是很想知道真正的解决办法和/或问题的根源。谢谢你的回答! - TheCrafter

0
最近我遇到了同样的问题,但是我找到了问题的根源。问题在于旧版和新版支持库之间的兼容性问题。似乎我在我的代码中使用了过时的东西和新的东西。
很抱歉我有点抽象,但这个问题已经四个月了,我记不清楚哪些代码行是错误的了。如果我没记错的话,问题出在了Android Studio为应用程序抽屉活动自动生成的方法上。我使用了Android Studio的抽屉应用程序项目模板,并选择支持非常旧的Android API,因此Android Studio选择了过时的Android Support Library。
关键是当我重构代码只使用非过时技术时,我解决了这个问题。
如果你正在与类似的问题作斗争,我强烈建议删除所有Android Studio(我假设你使用的是Android Studio或Eclipse)标记为过时的内容。

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接