Потоковое аудио с URL-адреса в Android с помощью MediaPlayer?

Я пытаюсь передать mp3 через http с помощью Android, встроенного в класс MediaPlayer. Документация подскажет мне, что это должно быть так же просто, как:

MediaPlayer mp = new MediaPlayer();
mp.setDataSource(URL_OF_FILE);
mp.prepare();
mp.start();

Однако я неоднократно получаю следующее. Я также пробовал разные URL-адреса. Пожалуйста, не говорите мне, что потоковая передача не работает на mp3.

E/PlayerDriver(   31): Command PLAYER_SET_DATA_SOURCE completed with an error or info PVMFErrNotSupported
W/PlayerDriver(   31): PVMFInfoErrorHandlingComplete
E/MediaPlayer(  198): error (1, -4)
E/MediaPlayer(  198): start called in state 0
E/MediaPlayer(  198): error (-38, 0)
E/MediaPlayer(  198): Error (1,-4)
E/MediaPlayer(  198): Error (-38,0)

Любая помощь очень ценится, спасибо S

Ответ 1

Простой медиаплеер с потоковым примером. Для части xml вам понадобится одна кнопка с кнопкой id1 и два изображения в вашей папке с именем button_pause и button_play и, пожалуйста, не забудьте добавить разрешение в Интернете в манифест.

public class MainActivity extends Activity {
private Button btn;
/**
 * help to toggle between play and pause.
 */
private boolean playPause;
private MediaPlayer mediaPlayer;
/**
 * remain false till media is not completed, inside OnCompletionListener make it true.
 */
private boolean intialStage = true;


@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    btn = (Button) findViewById(R.id.button1);
    mediaPlayer = new MediaPlayer();
    mediaPlayer.setAudioStreamType(AudioManager.STREAM_MUSIC);
    btn.setOnClickListener(pausePlay);

}

@Override
public boolean onCreateOptionsMenu(Menu menu) {
    // Inflate the menu; this adds items to the action bar if it is present.
    getMenuInflater().inflate(R.menu.activity_main, menu);
    return true;
}

private OnClickListener pausePlay = new OnClickListener() {

    @Override
    public void onClick(View v) {
        // TODO Auto-generated method stub
        // TODO Auto-generated method stub

        if (!playPause) {
            btn.setBackgroundResource(R.drawable.button_pause);
            if (intialStage)
                new Player()
                        .execute("http://www.virginmegastore.me/Library/Music/CD_001214/Tracks/Track1.mp3");
            else {
                if (!mediaPlayer.isPlaying())
                    mediaPlayer.start();
            }
            playPause = true;
        } else {
            btn.setBackgroundResource(R.drawable.button_play);
            if (mediaPlayer.isPlaying())
                mediaPlayer.pause();
            playPause = false;
        }
    }
};
/**
 * preparing mediaplayer will take sometime to buffer the content so prepare it inside the background thread and starting it on UI thread.
 * @author piyush
 *
 */

class Player extends AsyncTask<String, Void, Boolean> {
    private ProgressDialog progress;

    @Override
    protected Boolean doInBackground(String... params) {
        // TODO Auto-generated method stub
        Boolean prepared;
        try {

            mediaPlayer.setDataSource(params[0]);

            mediaPlayer.setOnCompletionListener(new OnCompletionListener() {

                @Override
                public void onCompletion(MediaPlayer mp) {
                    // TODO Auto-generated method stub
                    intialStage = true;
                    playPause=false;
                    btn.setBackgroundResource(R.drawable.button_play);
                    mediaPlayer.stop();
                    mediaPlayer.reset();
                }
            });
            mediaPlayer.prepare();
            prepared = true;
        } catch (IllegalArgumentException e) {
            // TODO Auto-generated catch block
            Log.d("IllegarArgument", e.getMessage());
            prepared = false;
            e.printStackTrace();
        } catch (SecurityException e) {
            // TODO Auto-generated catch block
            prepared = false;
            e.printStackTrace();
        } catch (IllegalStateException e) {
            // TODO Auto-generated catch block
            prepared = false;
            e.printStackTrace();
        } catch (IOException e) {
            // TODO Auto-generated catch block
            prepared = false;
            e.printStackTrace();
        }
        return prepared;
    }

    @Override
    protected void onPostExecute(Boolean result) {
        // TODO Auto-generated method stub
        super.onPostExecute(result);
        if (progress.isShowing()) {
            progress.cancel();
        }
        Log.d("Prepared", "//" + result);
        mediaPlayer.start();

        intialStage = false;
    }

    public Player() {
        progress = new ProgressDialog(MainActivity.this);
    }

    @Override
    protected void onPreExecute() {
        // TODO Auto-generated method stub
        super.onPreExecute();
        this.progress.setMessage("Buffering...");
        this.progress.show();

    }
}

@Override
protected void onPause() {
    // TODO Auto-generated method stub
    super.onPause();
    if (mediaPlayer != null) {
        mediaPlayer.reset();
        mediaPlayer.release();
        mediaPlayer = null;
    }
}

Ответ 2

Android MediaPlayer не поддерживает потоковое воспроизведение MP3 до 2.2. В более старых версиях ОС, похоже, только поток 3GP изначально. Вы можете попробовать код pocketjourney, хотя он старый (там новая версия здесь), и у меня были проблемы с его липкостью - он заикался всякий раз, когда он заправлялся буфер.

Приложение NPR News для Android является открытым исходным кодом и использует локальный прокси-сервер для обработки потоковой передачи MP3 в версиях ОС до 2.2. Вы можете увидеть соответствующий код в строках 199-216 (r94) здесь: http://code.google.com/p/npr-android-app/source/browse/Npr/src/org/npr/android/news/PlaybackService.java?r=7cf2352b5c3c0fbcdc18a5a8c67d836577e7e8e3

И это класс StreamProxy: http://code.google.com/p/npr-android-app/source/browse/Npr/src/org/npr/android/news/StreamProxy.java?r=e4984187f45c39a54ea6c88f71197762dbe10e72

Приложение NPR также по-прежнему получает "ошибку (-38, 0)" иногда во время потоковой передачи. Это может быть проблема с потоками или проблема смены сети. Проверьте отслеживатель ошибок для обновлений.

Ответ 3

Я предполагаю, что вы пытаетесь воспроизвести .pls прямо или что-то подобное.

попробуйте это:

1: код

mediaPlayer = MediaPlayer.create(this, Uri.parse("http://vprbbc.streamguys.net:80/vprbbc24.mp3"));
mediaPlayer.start();

2: файл .pls

Этот URL-адрес от BBC как пример. Это был файл .pls, который на linux я загрузил с помощью

wget http://foo.bar/file.pls

а затем я открыл vim (используйте ваш любимый редактор;), и я видел реальные URL-адреса внутри этого файла. К сожалению, не все из .pls являются обычным текстом.

Я читал, что 1.6 не поддерживает потоковое mp3 через http, но я только что протестировал код выше с Android 1.6 и 2.2 и не имел никаких проблем.

Удачи!

Ответ 4

У меня была такая же ошибка, как и у вас, и оказалось, что с кодом ничего не случилось. Проблема заключалась в том, что веб-сервер отправлял неправильный заголовок Content-Type.

Попробуйте прокси-сервер или что-то подобное, чтобы увидеть, какой тип содержимого передается веб-сервером.

Ответ 5

Используйте

 mediaplayer.setAudioStreamType(AudioManager.STREAM_MUSIC);
 mediaplayer.prepareAsync();
 mediaplayer.setOnPreparedListener(new MediaPlayer.OnPreparedListener() {
      @Override
      public void onPrepared(MediaPlayer mp) {
          mediaplayer.start();
      }
 });

Ответ 6

Нет вызова mp.start с OnPreparedListener, чтобы избежать нулевого состояния я журнала.

Ответ 7

Глядя на мои проекты:

  • https://github.com/master255/ImmortalPlayer поддержка http/FTP, один поток для чтения, отправки и сохранения данных кеша. Самый простой способ и самая быстрая работа. Сложная логика - лучший способ!
  • https://github.com/master255/VideoViewCache Простой Videoview с кешем. Два потока для воспроизведения и сохранения данных. Плохая логика, но если вам нужно, используйте это.