1

我有一個用戶(Samsung Galaxy S5,Android 4.4)報告的崩潰,我不明白髮生了什麼。這似乎是可以吸收的,但也許有些人遇到了同樣的問題,或類似的問題。MediaPlayer.setDataSource中的java.lang.IllegalStateException,使用鈴聲類

這裏是跟蹤:

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()是我的功能之一,並返回一個鈴聲對象是在活動的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 doc告訴我們,使用新的MediaPlayer()將其設置爲「空閒」狀態,而「空閒」狀態是調用setDataSource的正確狀態。 我真的不明白如何,顯然有一次在一千,我得到這個錯誤。我無法聯繫有錯誤的用戶,從未在我的終端上覆制它,所以我發現自己被阻止。 我正在考慮在我的代碼中捕獲錯誤,當它發生時,請使用我的Ringtone.setStreamType再次嘗試,因爲系統的瞬態「狀態」可能會在幾毫秒後改變。總之我很絕望:)

任何幫助或相似的經歷,將不勝感激:)

+1

錯誤來自您身邊的媒體播放器無法獲得正確格式的來源。你是否支持你的代碼中的所有格式。 – therealprashant 2014-10-17 08:52:22

+0

什麼是destroyLocalPlayer();? 以上評論可能是正確的,檢查支持的格式,由於您的操作系統版本.http://developer.android.com/guide/appendix/media-formats.html 也嘗試調試您的URL,硬編碼和嘗試與它 – Hunkeone 2014-10-17 09:23:04

+0

謝謝therealprashant和Hunkeone,我會試着朝這個方向努力,但對於我來說,數據源格式的問題導致出現IllegalStateException錯誤而不是IOException等並不合邏輯。 – jtag 2014-10-17 11:23:52

回答

9

就我而言,我不得不打電話reset()MediaPlayer例如,我有一個不同的URL提供它。

documentation而言,setDataSource僅在IDLE狀態下有效。

您還可以訪問this answer,使用圖很好地解釋。