MediaPlayer.setDataSource中出现java.lang.IllegalStateException错误,使用Ringtone类。

3

我收到了一个用户报告(三星Galaxy S5,Android 4.4),说他的应用程序崩溃了,但我不理解发生了什么。虽然似乎很难以置信,但或许有些人遇到了同样的问题,或者类似的问题。

以下是错误跟踪信息:

java.lang.RuntimeException: An error occured while executing doInBackground()
at android.os.AsyncTask$3.done(AsyncTask.java:300)
at java.util.concurrent.FutureTask.finishCompletion(FutureTask.java:355)
at java.util.concurrent.FutureTask.setException(FutureTask.java:222)
at java.util.concurrent.FutureTask.run(FutureTask.java:242)
at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1112)
at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:587)
at java.lang.Thread.run(Thread.java:841)
Caused by: java.lang.IllegalStateException
at android.media.MediaPlayer._setDataSource(Native Method)
at android.media.MediaPlayer.setDataSource(MediaPlayer.java:1383)
at android.media.MediaPlayer.setDataSource(MediaPlayer.java:1367)
at android.media.MediaPlayer.setDataSource(MediaPlayer.java:1302)
at android.media.MediaPlayer.setDataSource(MediaPlayer.java:1240)
at android.media.MediaPlayer.setDataSource(MediaPlayer.java:986)
at android.media.MediaPlayer.setDataSource(MediaPlayer.java:951)
at android.media.Ringtone.setUri(Ringtone.java:219)
at android.media.Ringtone.setStreamType(Ringtone.java:89)
at com.aasfet.clocklight.WakeActivity$RingAsyncTask.doInBackground(WakeActivity.java:510)
at com.aasfet.clocklight.WakeActivity$RingAsyncTask.doInBackground(WakeActivity.java:1)
at android.os.AsyncTask$2.call(AsyncTask.java:288)
at java.util.concurrent.FutureTask.run(FutureTask.java:237)
... 3 more

这是我的代码中出现错误的部分:

以下是出错的代码:

private class RingAsyncTask extends AsyncTask<Integer, Integer, Integer> {

        @Override
        protected Integer doInBackground(Integer... params) {
            int previousVolume = audioManager.getStreamVolume(AudioManager.STREAM_ALARM);
            int maxVolume = audioManager.getStreamMaxVolume(AudioManager.STREAM_ALARM);
            int newVolume = (int)(volume * (float)maxVolume);
            if(newVolume < 1){
                newVolume = 1;
            }
            if(progressive){
                audioManager.setStreamVolume(AudioManager.STREAM_ALARM, 1, 0);
            }else{
                audioManager.setStreamVolume(AudioManager.STREAM_ALARM, newVolume, 0);
            }
            getRingtone().setStreamType(AudioManager.STREAM_ALARM);
            getRingtone().play();
...

getRingtone() 是我的一个函数,返回一个 Ringtone 对象,该对象是在 activity 的 onResume 中通过 RingtoneManager 游标获取的。我认为那里没有任何问题。

在 Android 源码中,Ringtone.setStreamType 调用了 Ringtone.setURI。下面是 setURI 的代码:

 public void setUri(Uri uri) {
168        destroyLocalPlayer();
169
170        mUri = uri;
171        if (mUri == null) {
172            return;
173        }
174
175        // TODO: detect READ_EXTERNAL and specific content provider case, instead of relying on throwing
176
177        // try opening uri locally before delegating to remote player
178        mLocalPlayer = new MediaPlayer();
179        try {
180            mLocalPlayer.setDataSource(mContext, mUri);
181            mLocalPlayer.setAudioStreamType(mStreamType);
182            mLocalPlayer.prepare();
...

因此,setURI会创建一个新的MediaPlayer,然后在MediaPlayer上调用setDataSource。Android文档告诉我们,使用new MediaPlayer()将其设置为“空闲”状态,并且“空闲”状态是调用setDataSource的正确状态。

我真的不明白,为什么偶尔会出现这个错误。我无法联系到遇到该错误的用户,也从未在我的终端重现过该错误,所以我被阻塞了。

我想在我的代码中捕获错误,并在发生错误时尝试使用Ringtone.setStreamType来再次尝试,假设它发生是因为系统的瞬态“状态”可能已经在几毫秒后发生了变化。总之,我非常绝望 :)

如果有任何帮助或类似的经验,将不胜感激 :)


1
错误是你这边的问题,媒体播放器无法获取正确格式的源文件。你的代码是否支持所有格式? - undefined
destroyLocalPlayer();的作用是什么? 上面的评论可能是正确的,请检查由于您的操作系统版本原因支持的格式。http://developer.android.com/guide/appendix/media-formats.html 同时尝试调试您的URL,将一个硬编码进去并尝试使用它。 - undefined
谢谢therealprashant和Hunkeone,我会尝试这个方向,但是对我来说,数据源格式的问题导致IllegalStateException错误,而不是IOException,这似乎不太合理。 - undefined
1个回答

14

在我的情况下,我必须在给MediaPlayer实例提供不同的URL之前调用reset()

根据文档,setDataSource仅在IDLE状态下有效。

你也可以访问这个答案,用图解释得非常清楚明了。


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