1

사용자 (Samsung Galaxy S5, Android 4.4)에서 충돌이보고되었으며 무슨 일이 일어나고 있는지 이해할 수 없습니다. 그것은 imporbable하지만 어쩌면 어떤 사람들은 같은 문제가 발생하거나 비슷한 것으로 보인다. 여기 Ringtone 클래스를 사용하는 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

+0

destroyLocalPlayer();는 무엇입니까? 위의 위의 설명이 맞을 수도 있습니다. OS 버전에 따라 지원되는 형식을 확인하십시오 .http : //developer.android.com/guide/appendix/media-formats.html URL을 디버깅하고 하드 코드 한 다음 시도해보십시오. – Hunkeone

+0

therealprashant와 Hunkeone을 고맙다. 나는이 방향으로 시도 하겠지만 데이터 소스 형식의 문제로 인해 IOException이 아닌 IllegalStateException 오류가 발생한다는 것이 논리적으로 들리지 않는다. – jtag

답변

9

, 내가 다른 URL로 공급하기 전에 MediaPlayer 인스턴스 reset()를 호출했다.

documentation이가는 한 setDataSource는 유휴 상태에서만 유효합니다.

this answer을 방문하여 다이어그램을 사용하여 매우 잘 설명 할 수도 있습니다.

관련 문제