1

(我已经解决了,只需向下滚动直到 EDIT 2)

我有一个应用程序,在打开时会循环播放背景音乐。我决定我可以更进一步,在应用程序暂停时暂停 bgm,并在恢复时暂停的地方播放。它工作正常。

但是,我遇到的问题是退出应用程序时。它给出了错误: 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 已经发布并且无法完成命令backgroundMusic.pause()

我通过创建另一个全局变量来解决这个问题: boolean flag = false;

该标志检查是否选择了退出按钮以及用户是否想要退出游戏。一旦用户想要退出游戏,它就会将该标志设置为真。

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

我将标志设置为true,以便当它从finish()进入onPause()方法时(关于Android的活动生命周期我可能是错误的,但似乎onPause()在finish()之内或之后执行),这将执行:

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

}

在这里,如果用户想退出游戏,它只会执行 super.onPause()。否则,如果游戏真的被暂停(通过主页按钮或其他东西),那么 Flag 仍然是 false 并且背景音乐将被暂停。

我还修改了 onResume() 以确保安全:

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

1 回答 1

1

根据 Android 文档:

“如果内部播放器引擎尚未初始化或已释放,则出现 IllegalStateException。”

我想说首先确保你已经初始化和/或没有释放播放器。

您还需要在 onPause() 中释放播放器,除非您有非常特殊的要求来保留它。

此外,如果您为 MediaPLayer 使用静态全局变量,这可能是问题所在。仔细检查它是如何被初始化和释放的,尤其是在类和类似的东西之间切换时。

如果您确实需要保存状态,那么您可以使用getCurrentPosition()onPause() 和 onResume() 中的方法保存轨道的当前位置,您可以使用 seekTo (int msec)方法并传入从 getCUrrentPosition() 获得的值,以便您可以从中断的地方继续播放。

这些功能的官方文档在这里这里

于 2012-10-07T16:19:30.337 回答