1.错误log
java.lang.IllegalStateException
at android.media.MediaPlayer._setDataSource(Native Method)
at android.media.MediaPlayer.setDataSource(MediaPlayer.java:1157)
at android.media.MediaPlayer.setDataSource(MediaPlayer.java:1140)
at android.media.MediaPlayer.setDataSource(MediaPlayer.java:1094)
at android.media.MediaPlayer.setDataSource(MediaPlayer.java:1043)
at com.stone.musicplayer.service.MusicService$MyBinder.initMediaPlayerFile(MusicService.java:195)
at com.stone.musicplayer.service.MusicService$MyBinder.playOrPause(MusicService.java:107)
at com.stone.musicplayer.PlayMusicActivity.onClick(PlayMusicActivity.java:247)
at android.view.View.performClick(View.java:5647)
at android.view.View$PerformClick.run(View.java:22465)
at android.os.Handler.handleCallback(Handler.java:754)
at android.os.Handler.dispatchMessage(Handler.java:95)
at android.os.Looper.loop(Looper.java:163)
at android.app.ActivityThread.main(ActivityThread.java:6394)
at java.lang.reflect.Method.invoke(Native Method)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:904)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:794)
2.错误原因
IllegalStateException这个异常它是指“非法的状态”。
因为android的MediaRecorder 和 MediaPlayer (MediaRecorder ,MediaPlayer 都是用c++实现的)API中用到了JNI。
出现这个异常,大部分原因是因为java里面的MediaRecorder /MediaPlayer 对象和native的对象不一致导致的。
java会分别在堆内存和栈内存中操作,堆中放内容,栈中放地址,栈中的地址指向堆中内容。
在java中释放掉MediaRecorder /MediaPlayer的对象时,释放的是jni的对象,而java中只释放了栈中的内容,而堆中的内容没有释放。说到底就是没释放干净,导致出现对象不一致。
3.出现场景
1.比如想在生命周期的onDestroy中释放,向这样
@Override
public void onDestroy() {
super.onDestroy();
if (MediaPlayUtil.getmMediaPlayer() != null) {
if (MediaPlayUtil.getmMediaPlayer().isPlaying())
MediaPlayUtil.getmMediaPlayer().stop();
MediaPlayUtil.getmMediaPlayer().reset();
MediaPlayUtil.getmMediaPlayer().release();
}
//不加下面这句,就可能释放不干净
//System.exit(0);
}
释放阶段的问题,可能是你没有写
System.exit(0);
2.还有一种可能
切换歌曲时,没有调用reset()方法,没有会报IllegalStateException