MediaPlayer.setDataSource中的java.lang.IllegalStateException,使用Ringtone类

时间:2014-10-17 08:22:07

标签: android android-mediaplayer illegalstateexception ringtone

我的用户报告了崩溃(三星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对象,该对象是在活动的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文件告诉我们使用新的MediaPlayer()将其设置为“空闲”状态,并且“空闲”状态是调用setDataSource的正确状态。 我真的不明白如何,显然有一次,我得到这个错误。我无法联系有错误的用户,从来没有在我的身上复制它,所以我发现自己被阻止了。 我正在考虑在我的代码中捕获错误,当它发生时,再次尝试使用我的Ringtone.setStreamType,假设它发生的原因是系统的瞬态“状态”可能在几毫秒之后发生了变化。总之,我非常绝望:)。

任何帮助或类似经历将不胜感激:)

1 个答案:

答案 0 :(得分:11)

在我的情况下,我必须在reset()实例上调用MediaPlayer,然后才提供不同的网址。

documentation而言,setDataSource仅在IDLE状态下有效。

您还可以访问this answer,使用图表进行了非常好的解释。