21

我需要播放实时音频流,实际上是广播。问题是我还需要管理 20 分钟的流媒体缓冲区。据我了解,用android实现并不容易。

首先我检查了 MediaPlayer,但它没有提供任何缓冲区管理方法。事实上,你甚至不能直接设置缓冲区大小。

其次,我尝试使用本地文件管理缓冲区:逐步将流下载到临时文件并切换它们。但是当您想切换到以下文件时(在 MediaPlayer 中更改数据源),音频不会连续播放。您可以听到短暂的中断。

最后一个想法是使用流代理。通常它用于在 8 以下的 Android 版本中播放流。在流代理中,您创建 ServerSocket,从音频流中读取并写入播放器。所以实际上我可以在那里管理缓冲。我可以随心所欲地缓存流并写入 MediaPlayer。但。它不适用于 Android 8。

我遇到异常:对等方重置连接 java.net.SocketException:对等方重置连接。MediaPlayer 8 不想从套接字读取数据。

因此我有两个问题:1)还有什么其他方法可以实现流缓冲?2) 如何为 android 8 适配 StreamProxy?

任何想法表示赞赏。

谢谢

4

5 回答 5

8

我使用与 NPR 项目相同的 StreamProxy - https://code.google.com/p/npr-android-app/source/browse/Npr/src/org/npr/android/news/StreamProxy.java

所以它得到原始音频流:

  String url = request.getRequestLine().getUri();
  HttpResponse realResponse = download(url);
  ...
  InputStream data = realResponse.getEntity().getContent();

并从此流写入客户端套接字:

  byte[] buff = new byte[1024 * 50];
  while (isRunning && (readBytes = data.read(buff, 0, buff.length)) != -1) {
    client.getOutputStream().write(buff, 0, readBytes);
  }

(您可以从上面的链接中获取整个代码。)

最后是他们如何初始化播放器(PlaybackService):

  if (stream && sdkVersion < 8) {
    if (proxy == null) {
      proxy = new StreamProxy();
      proxy.init();
      proxy.start();
    }
    String proxyUrl = String.format("http://127.0.0.1:%d/%s", proxy.getPort(), url);
    playUrl = proxyUrl;
   }
  ...
  mediaPlayer.setDataSource(playUrl);
  mediaPlayer.setAudioStreamType(AudioManager.STREAM_MUSIC);
  mediaPlayer.prepareAsync();

所以他们检查 SDK 版本。但是,如果我省略此检查并使用 SDK 8 的代理,我也会遇到异常。奇怪的是 MediaPlayer 甚至不尝试读取流:

12-30 15:09:41.576: DEBUG/StreamProxy(266): downloading...
12-30 15:09:41.597: DEBUG/StreamProxy(266): reading headers
12-30 15:09:41.597: DEBUG/StreamProxy(266): headers done
12-30 15:09:41.647: DEBUG/StreamProxy(266): writing to client
12-30 15:09:41.857: INFO/AwesomePlayer(34): mConnectingDataSource->connect() returned -    1007
12-30 15:09:41.857: ERROR/MediaPlayer(266): error (1, -1007)
12-30 15:09:41.867: ERROR/MediaPlayer(266): Error (1,-1007)
12-30 15:09:41.867: WARN/AudioService(266): onError(1, -1007)
12-30 15:09:41.867: WARN/AudioService(266): MediaPlayer refused to play current item.  Bailing on prepare.
12-30 15:09:41.867: WARN/AudioService(266): onComplete()
12-30 15:09:42.097: ERROR/StreamProxy(266): Connection reset by peer
        java.net.SocketException: Connection reset by peer
        at org.apache.harmony.luni.platform.OSNetworkSystem.writeSocketImpl(Native Method)
        at org.apache.harmony.luni.platform.OSNetworkSystem.write(OSNetworkSystem.java:723)
        at org.apache.harmony.luni.net.PlainSocketImpl.write(PlainSocketImpl.java:578)
        at org.apache.harmony.luni.net.SocketOutputStream.write(SocketOutputStream.java:59)
        at com.skyblue.service.media.StreamProxy.processRequest(StreamProxy.java:204)
        at com.skyblue.service.media.StreamProxy.run(StreamProxy.java:103)
        at java.lang.Thread.run(Thread.java:1096)

似乎 MediaPlayer 变得更聪明了。如果我通过这样的 url "http://127.0.0.1:%d/%s",它不仅要获得字节,还要获得“完整”的 http 响应。

另外我想知道是否还有其他方法可以实现缓冲?据我所知,MediaPlayer 只消耗文件和 url。文件解决方案不起作用。所以我必须使用套接字进行流传输。

谢谢

于 2010-12-30T15:23:51.877 回答
5

我刚刚测试了以下修复程序,它可以工作。此问题是由. 将其更改为“\r\n”应该会使错误消失。processRequest()StreamProxy

至于实现自定义流式传输,我过去曾使用过它,尽管它似乎主要用于无限流式广播。 http://blog.pocketjourney.com/2009/12/27/android-streaming-mediaplayer-tutorial-updated-to-v1-5-cupcake/

于 2011-06-08T22:12:42.140 回答
3

当您寻找或跳过或连接丢失并且 MediaPlayer 不断重新连接到代理服务器时,您必须在从客户端获得请求和 range(int) 后发送状态为 206 的此响应。

String headers += "HTTP/1.1 206 Partial Content\r\n";
headers += "Content-Type: audio/mpeg\r\n";
headers += "Accept-Ranges: bytes\r\n";
headers += "Content-Length: " + (fileSize-range) + "\r\n";
headers += "Content-Range: bytes "+range + "-" + fileSize + "/*\r\n";
headers += "\r\n";

当您从 MediaPlayer 收到 HTTP 标头中不包含 Range 的请求时,它正在请求一个新的流文件,在这种情况下,您的响应标头应如下所示:

String headers = "HTTP/1.1 200 OK\r\n";
headers += "Content-Type: audio/mpeg\r\n";
headers += "Accept-Ranges: bytes\r\n";
headers += "Content-Length: " + fileSize + "\r\n";
headers += "\r\n";

享受!

于 2013-01-21T04:06:45.677 回答
2

SDK 8 中的 Mediaplayer 将无法读取代理 URL。但根据我目前的工作,这因设备而异。在我的三星 ACE (SDK 8) 中,代理连接工作正常,但在我的 HTC Incredible S 中,代理连接会出现与您相同的问题。直接连接到音频流可以正常工作,但这会导致某些设备(如 sprint 上的 EVO)出现故障。

你得到这个问题的解决方案了吗?你是怎么处理的?

-哈里

于 2011-06-06T04:08:50.680 回答
1

我意识到这个问题已经有 5 年历史了,但万一其他人在徘徊:ExoPlayer 是谷歌的库,它让您可以更好地控制缓冲区大小等选项。

    //DefaultUriDataSource – For playing media that can be either local or loaded over the network.
    DefaultUriDataSource dataSource = new DefaultUriDataSource(WgtechApplication.getAppContext(),
            Util.getUserAgent(WgtechApplication.getAppContext(), WgtechApplication.class.getSimpleName()));
    Allocator allocator = new DefaultAllocator(BUFFER_SEGMENT_SIZE);

    //ExtractorSampleSource – For formats such as MP3, M4A, MP4, WebM, MPEG-TS and AAC.
    ExtractorSampleSource sampleSource = new ExtractorSampleSource(Uri.parse(RADIO_STREAMING_URL),
            dataSource, allocator, BUFFER_SEGMENT_COUNT * BUFFER_SEGMENT_SIZE);
    player.prepare(new MediaCodecAudioTrackRenderer(sampleSource));

这是我创建的一个小项目来学习如何使用它。 https://github.com/feresr/MyMediaPlayer

http://developer.android.com/guide/topics/media/exoplayer.html https://github.com/google/ExoPlayer

于 2016-01-22T05:22:29.573 回答