Невозможно приостановить действие — исключение из-за недопустимого состояния при воспроизведении BGM на Android

(Я уже решил это, просто прокрутите вниз до РЕДАКТИРОВАТЬ 2)

У меня есть приложение, которое при открытии воспроизводит фоновую музыку в цикле. Я решил, что могу пойти еще дальше и приостановить фоновую музыку, когда приложение приостановлено, и воспроизвести его там, где оно было приостановлено при возобновлении. Это работает нормально.

Однако проблема, с которой я сталкиваюсь, заключается в том, что я выхожу из приложения. Выдает ошибку: 10-07 22:50:04.535: E/AndroidRuntime(5852): java.lang.RuntimeException: Unable to pause activity {com.cs119.megamanstrikes/com.cs119.megamanstrikes.MainActivity}: java.lang.IllegalStateException

Мой onCreate выглядит так:

@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    backgroundMusic = MediaPlayer.create(MainActivity.this, R.raw.mmbn_3);
    backgroundMusic.setLooping(true);
    backgroundMusic.start();

    Button start = (Button) findViewById(R.id.Start);
    Button highScore = (Button) findViewById(R.id.highScore);
    Button quit = (Button) findViewById(R.id.quit);

    start.setOnClickListener(new OnClickListener() {
        @Override
        public void onClick(View v){
            startGame();
        }
    });

    highScore.setOnClickListener(new OnClickListener() {
        @Override
        public void onClick(View v){
            highScore();
        }
    });

    quit.setOnClickListener(new OnClickListener() {
        @Override
        public void onClick(View v){
            quit();
        }
    });


}

Это переопределения для onPause и onResume:

@Override 
public void onPause(){
   super.onPause();
   backgroundMusic.pause();
}

@Override
public void onResume(){
    super.onResume();
    backgroundMusic.start();
}

И когда пользователь нажимает кнопку выхода или кнопку возврата:

@Override
public void onBackPressed(){
    quit();
}

public void quit(){
    quitDialog();
}

private void quitDialog() {
    AlertDialog.Builder alertDialogBuilder = new AlertDialog.Builder(this);

    // Set title.
    alertDialogBuilder.setTitle("QUIT");

    // Set dialog message.
    alertDialogBuilder
            .setMessage("Are you sure you want to quit?")
            .setCancelable(false)
            .setPositiveButton("Yes",
                    new DialogInterface.OnClickListener() {
                        public void onClick(DialogInterface dialog, int id) {
                            //backgroundMusic.stop();
                            backgroundMusic.release();
                            finish();
                        }
                    })
            .setNegativeButton("No", new DialogInterface.OnClickListener() {
                public void onClick(DialogInterface dialog, int id) {
                    // if this button is clicked, resume teh gaem
                    //inputName();
                }
            });

    // Create alert dialog.
    AlertDialog alertDialog = alertDialogBuilder.create();

    // Show it.
    alertDialog.show();
}

После нажатия «Да» для выхода из игры приложение зависает и принудительно закрывается. Вот полная ошибка LogCat

10-07 22:50:04.535: E/AndroidRuntime(5852): FATAL EXCEPTION: main
10-07 22:50:04.535: E/AndroidRuntime(5852): java.lang.RuntimeException: Unable to pause activity {com.cs119.megamanstrikescom.cs119.megamanstrikes.MainActivity}:java.lang.IllegalStateException
10-07 22:50:04.535: E/AndroidRuntime(5852): at android.app.ActivityThread.performPauseActivity(ActivityThread.java:2358)
10-07 22:50:04.535: E/AndroidRuntime(5852): at android.app.ActivityThread.performPauseActivity(ActivityThread.java:2315)
10-07 22:50:04.535: E/AndroidRuntime(5852): at android.app.ActivityThread.handlePauseActivity(ActivityThread.java:2295)
10-07 22:50:04.535: E/AndroidRuntime(5852): at android.app.ActivityThread.access$1700(ActivityThread.java:117)
10-07 22:50:04.535: E/AndroidRuntime(5852): at android.app.ActivityThread$H.handleMessage(ActivityThread.java:946)
10-07 22:50:04.535: E/AndroidRuntime(5852): at android.os.Handler.dispatchMessage(Handler.java:99)
10-07 22:50:04.535: E/AndroidRuntime(5852): at android.os.Looper.loop(Looper.java:123)
10-07 22:50:04.535: E/AndroidRuntime(5852): at android.app.ActivityThread.main(ActivityThread.java:3691)
10-07 22:50:04.535: E/AndroidRuntime(5852): at java.lang.reflect.Method.invokeNative(Native Method)
10-07 22:50:04.535: E/AndroidRuntime(5852): at java.lang.reflect.Method.invoke(Method.java:507)
10-07 22:50:04.535: E/AndroidRuntime(5852): at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:847)
10-07 22:50:04.535: E/AndroidRuntime(5852): at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:605)
10-07 22:50:04.535: E/AndroidRuntime(5852): at dalvik.system.NativeStart.main(Native Method)
10-07 22:50:04.535: E/AndroidRuntime(5852): Caused by: java.lang.IllegalStateException
10-07 22:50:04.535: E/AndroidRuntime(5852): at android.media.MediaPlayer._pause(Native Method)
10-07 22:50:04.535: E/AndroidRuntime(5852): at android.media.MediaPlayer.pause(MediaPlayer.java:1001)
10-07 22:50:04.535: E/AndroidRuntime(5852): at com.cs119.megamanstrikes.MainActivity.onPause(MainActivity.java:163)
10-07 22:50:04.535: E/AndroidRuntime(5852): at android.app.Activity.performPause(Activity.java:3877)
10-07 22:50:04.535: E/AndroidRuntime(5852): at android.app.Instrumentation.callActivityOnPause(Instrumentation.java:1191)
10-07 22:50:04.535: E/AndroidRuntime(5852): at android.app.ActivityThread.performPauseActivity(ActivityThread.java:2345)
10-07 22:50:04.535: E/AndroidRuntime(5852): ... 12 more

Я также пытался сделать backgroundMusic.stop(); вместо backgroundMusic.pause() в методе onPause, однако при возобновлении работы приложения звук не воспроизводится, даже если backgroundMusic.play() присутствует в методе onResume().

Я также заметил, что ошибка все еще существует независимо от того, что я помещаю в onPause() и onResume(), и я пытаюсь выйти из приложения.

Спасибо за помощь.

РЕДАКТИРОВАТЬ 1: я размещаю строки system.out в переопределениях onPause и onResume, чтобы я мог видеть, когда он выполняется. При запуске приложения выполняется onResume, а при закрытии приложения выполняется onPause. Я очень смущен тем, почему это происходит, поскольку onResume отличается от onCreate.

РЕДАКТИРОВАТЬ 2: Хорошо, я решил проблему. Проблема в основном возникает из-за кнопки выхода, поскольку она выполняет следующие строки кода:

backgroundMusic.stop();
backgroundMusic.release();
finish();

Каким-то образом он переходит к методу onPause и пытается выполнить backgroundMusic.pause(). Теперь приложение принудительно закрывается, потому что фоновая музыка уже выпущена и не может завершить команду backgroundMusic.pause().

Я решил эту проблему, создав еще одну глобальную переменную: boolean flag = false;

Этот флаг проверяет, была ли нажата кнопка выхода и хочет ли пользователь выйти из игры. Как только пользователь захочет выйти из игры, он установит флаг в значение true.

flag = true;
backgroundMusic.stop();
backgroundMusic.release();
finish();

Я устанавливаю флаг в значение true, чтобы при переходе к методу onPause() из finish() (может быть, я ошибаюсь в отношении жизненного цикла активности Android, но кажется, что onPause() выполняется внутри или после finish() ), это выполнит:

@Override 
public void onPause(){
    super.onPause();
    if(!flag){
        if(backgroundMusic.isPlaying()){
            backgroundMusic.pause();
        }
    }

}

Здесь он будет выполнять super.onPause() только в том случае, если пользователь хочет выйти из игры. В противном случае, если бы игра действительно была поставлена ​​на паузу (кнопкой домой или чем-то еще), то Флаг все равно был бы ложным, и фоновая музыка была бы поставлена ​​на паузу.

Я также изменил onResume(), чтобы быть в безопасности:

@Override
public void onResume(){
    super.onResume();
    if(!backgroundMusic.isPlaying()){
        backgroundMusic.start();
    }
}

person Razgriz    schedule 07.10.2012    source источник
comment
Я бы проверил перед паузой и при возобновлении, если (backgroundMusic!=null) и, возможно, также backgrounmusic.isplaying... Просто догадываюсь, но попробуйте изменить порядок в методе onpause. Сначала пауза, а затем super.onPause.   -  person Martin Vandzura    schedule 07.10.2012
comment
Я тоже получил этот сбой на onPause. И добавь флаг как у тебя, и будет надежда, что краш пропал =) Спасибо   -  person CReaTuS    schedule 28.05.2013


Ответы (1)


Согласно документам Android:

«IllegalStateException, если внутренний движок проигрывателя не был инициализирован или выпущен».

Я бы сказал, сначала убедитесь, что вы инициализировали и/или не выпустили плеер.

вам также нужно отпустить игрока в onPause(), если у вас нет особых требований, чтобы сохранить его.

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

Если вам действительно нужно сохранить состояние, вы можете сохранить текущую позицию дорожки, используя метод getCurrentPosition() в вашем onPause(), а в вашем onResume() вы можете использовать метод seekTo (int msec) и передать значение, полученное из getCUrrentPosition(), чтобы Вы можете возобновить воспроизведение с того места, на котором остановились.

официальные документы по этим функциям находятся здесь и здесь

person Anup Cowkur    schedule 07.10.2012
comment
Мне нужно сохранить фоновую музыку в onPause(), чтобы я мог воспроизвести ее с того места, где она остановилась в методе onResume(). Я думаю, что решил проблему, связанную с моим методом onPause(). - person Razgriz; 08.10.2012