Сбой приложения onResume при отмене уведомления

Я пытаюсь отменить свое уведомление в onResume(), но оно вылетает:

С помощью displayNotification() я создаю уведомление. Я также попытался установить cancelNotification() в попытке поймать. Но это не решает проблему. Но без него я даже не могу запустить приложение для создания уведомления.

Вот мои Codesnippets:

OnResume:

@Override
    protected void onResume() //activity was resumed and is visible again 
    {
        Log.d(logtag,"onResume() called");
        super.onResume();
        cancelNotification();
    }

отменитьУведомление():

protected void cancelNotification() 
    {
        Log.i("Cancel", "notification");
        mNotificationManager.cancel(1);
    }

отображениеУведомление():

protected void displayNotification(String message, String ticker) 
    {
        Log.i("Start", "notification");

        /* Invoking the default notification service */
        NotificationCompat.Builder  mBuilder = 
        new NotificationCompat.Builder(this);   

        mBuilder.setContentTitle("Neue Nachricht!");
        mBuilder.setContentText(message);
        mBuilder.setTicker(ticker);
        mBuilder.setSmallIcon(R.drawable.ic_launcher);

        /* Increase notification number every time a new notification arrives */
        mBuilder.setNumber(++numMessages);

        /* Creates an explicit intent for an Activity in your app */
        Intent resultIntent = new Intent(this, MainActivity.class);

        TaskStackBuilder stackBuilder = TaskStackBuilder.create(this);
        stackBuilder.addParentStack(MainActivity.class);

        /* Adds the Intent that starts the Activity to the top of the stack */
        stackBuilder.addNextIntent(resultIntent);
        PendingIntent resultPendingIntent =
        stackBuilder.getPendingIntent(
            0,
            PendingIntent.FLAG_UPDATE_CURRENT
        );

        mBuilder.setContentIntent(resultPendingIntent);

        mNotificationManager = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);

        /* notificationID allows you to update the notification later on. */
        mNotificationManager.notify(1, mBuilder.build());
    }

Ошибка:

10-23 17:27:35.763: E/AndroidRuntime(349): FATAL EXCEPTION: main
10-23 17:27:35.763: E/AndroidRuntime(349): java.lang.RuntimeException: Unable to resume activity {com.example.blauzahn/com.example.blauzahn.MainActivity}: java.lang.NullPointerException
10-23 17:27:35.763: E/AndroidRuntime(349):  at android.app.ActivityThread.performResumeActivity(ActivityThread.java:2120)
10-23 17:27:35.763: E/AndroidRuntime(349):  at android.app.ActivityThread.handleResumeActivity(ActivityThread.java:2135)
10-23 17:27:35.763: E/AndroidRuntime(349):  at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:1668)
10-23 17:27:35.763: E/AndroidRuntime(349):  at android.app.ActivityThread.access$1500(ActivityThread.java:117)
10-23 17:27:35.763: E/AndroidRuntime(349):  at android.app.ActivityThread$H.handleMessage(ActivityThread.java:931)
10-23 17:27:35.763: E/AndroidRuntime(349):  at android.os.Handler.dispatchMessage(Handler.java:99)
10-23 17:27:35.763: E/AndroidRuntime(349):  at android.os.Looper.loop(Looper.java:123)
10-23 17:27:35.763: E/AndroidRuntime(349):  at android.app.ActivityThread.main(ActivityThread.java:3683)
10-23 17:27:35.763: E/AndroidRuntime(349):  at java.lang.reflect.Method.invokeNative(Native Method)
10-23 17:27:35.763: E/AndroidRuntime(349):  at java.lang.reflect.Method.invoke(Method.java:507)
10-23 17:27:35.763: E/AndroidRuntime(349):  at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:839)
10-23 17:27:35.763: E/AndroidRuntime(349):  at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:597)
10-23 17:27:35.763: E/AndroidRuntime(349):  at dalvik.system.NativeStart.main(Native Method)
10-23 17:27:35.763: E/AndroidRuntime(349): Caused by: java.lang.NullPointerException
10-23 17:27:35.763: E/AndroidRuntime(349):  at com.example.blauzahn.MainActivity.cancelNotification(MainActivity.java:388)
10-23 17:27:35.763: E/AndroidRuntime(349):  at com.example.blauzahn.MainActivity.onResume(MainActivity.java:976)
10-23 17:27:35.763: E/AndroidRuntime(349):  at android.app.Instrumentation.callActivityOnResume(Instrumentation.java:1150)
10-23 17:27:35.763: E/AndroidRuntime(349):  at android.app.Activity.performResume(Activity.java:3832)
10-23 17:27:35.763: E/AndroidRuntime(349):  at android.app.ActivityThread.performResumeActivity(ActivityThread.java:2110)
10-23 17:27:35.763: E/AndroidRuntime(349):  ... 12 more

Какие-либо предложения?


person progNewbie    schedule 23.10.2014    source источник
comment
Почему бы не отменить onPause()?   -  person greenapps    schedule 23.10.2014
comment
потому что уведомление вызывается, когда активность даже не видна.   -  person progNewbie    schedule 23.10.2014


Ответы (2)


mNotificationManager равно нулю в cancelNotification. Просто повторите его:

protected void cancelNotification() {
    Log.i("Cancel", "notification");
    mNotificationManager = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
    mNotificationManager.cancel(1);
}
person Egor Neliuba    schedule 23.10.2014

@EgnorN отвечает прямо в цель: mNotificationManager имеет значение null в cancelNotification().

Чтобы добавить несколько более широких уроков:

  1. Мы можем отлаживать такие проблемы из трассировки стека:
    Caused by: java.lang.NullPointerException
      at ...cancelNotification(MainActivity.java:388)
      at ...onResume(MainActivity.java:976)

Это указывает на то, что программа столкнулась с нулевым значением в cancelNotification() в MainActivity.java в строке 388. Это должно быть:

    mNotificationManager.cancel(1);

mNotificationManager должно быть нулевым, чтобы вызвать это.

Если в данной строке есть несколько вещей, которые могут быть нулевыми, можно использовать точку останова отладчика, чтобы остановиться на этой строке и посмотреть на разные значения. В качестве альтернативы разбейте его на несколько строк и запустите снова, чтобы увидеть, какая строка вызывает исключение.

  1. Если вы измените код, чтобы получить NotificationManager в каждом методе, который в нем нуждается, и хранить его в локальной переменной, а не в переменной экземпляра, это позволит избежать категории подобных проблем, когда код ошибочно предполагает, что переменные экземпляра действия еще поставил.

В любом случае ознакомьтесь с жизненным циклом активности. Важно знать, когда экземпляр Activity создается и удаляется.

  1. Если вы создаете свой проект с помощью версии API 21 библиотеки support.v4.app, вы можете сделать свой код немного проще, более безопасным для типов и более обратно совместимым, используя недавно добавленный метод NotificationManagerCompat.from(Context) для получения NotificationManagerCompat, который похож на NotificationManager плюс запасные варианты для старых платформ.
person Jerry101    schedule 23.10.2014
comment
у меня отключены и включены некоторые кнопки, прежде чем я сверну свое приложение. При нажатии на кнопку приложения в меню приложение возобновляет работу, и кнопки становятся такими, как я их установил. Но при нажатии на уведомление на панели действий приложение открывается с набором кнопок по умолчанию. Вы знаете, как предотвратить это? - person progNewbie; 24.10.2014
comment
Это новый вопрос, но, вкратце, состояние действия зависит от того, как вы создаете PendingIntent для перехода туда из уведомления (см., например, developer.android.com/training/notify-user/), сохраняет ли действие и восстанавливает состояние (см., например, developer.android.com/training/basics/activity-lifecycle/) и, возможно, другие факторы. - person Jerry101; 24.10.2014