VideoView Stream加载不够缓冲

时间:2016-10-13 16:39:44

标签: android stream android-mediaplayer android-videoview

我的Android应用在VideoView中在线播放视频。从文件播放视频时,它可以正常工作,甚至直播(m3u8);它总是来自同一个源,当我使用外部播放器/浏览器时,它同样流畅(所以我不认为这是源的问题,这是文件的变体像这样:https://publish.dvlabs.com/democracynow/360/dn2016-0810.mp4

Android Monitor会在崩溃前记录下来:

10-13 12:02:56.204 32460-32748/com.workingagenda.democracydroid D/MediaHTTPConnection: filterOutInternalHeaders: key=User-Agent, val= stagefright/1.2 (Linux;Android 6.0.1)
10-13 12:02:56.205 32460-32472/com.workingagenda.democracydroid D/MediaHTTPConnection: proxy null port 0
10-13 12:02:57.904 32460-32460/com.workingagenda.democracydroid D/MediaPlayer: getMetadata
10-13 12:02:58.438 32460-377/com.workingagenda.democracydroid W/MediaPlayer: info/warning (3, 0)

然后我崩溃时会收到这些日志:

10-13 12:05:33.812 32460-32472/com.workingagenda.democracydroid W/MediaHTTPConnection: readAt 26869519 / 241 => java.net.ProtocolException: unexpected end of stream
10-13 12:08:32.480 32460-3546/com.workingagenda.democracydroid E/MediaPlayer: error (1, -1004)
10-13 12:08:32.480 32460-32460/com.workingagenda.democracydroid E/MediaPlayer: Error (1,-1004)
10-13 12:08:32.481 32460-32460/com.workingagenda.democracydroid D/VideoView: Error: 1,-1004

                                                                         [ 10-13 12:08:32.512  5066:  453 E/         ]
                                                                         Destroy C2D instance

                                                                         [ 10-13 12:08:32.512  5066:  453 E/         ]
                                                                         Destroy C2D instance
10-13 12:08:32.635 32460-32472/com.workingagenda.democracydroid E/MediaPlayer: error (1, -1004)
10-13 12:08:32.668 32460-32460/com.workingagenda.democracydroid E/MediaPlayer: Error (1,-1004)
10-13 12:08:32.668 32460-32460/com.workingagenda.democracydroid D/VideoView: Error: 1,-1004

更确切地说我的问题:

  1. 我想知道这个错误E/MediaPlayer: Error (1,-1004)是什么(因为我还没有在网上找到任何关于它的信息)。
  2. 如果这是我怀疑的,基本上是文件/流错误的结束,那么我希望得到一些帮助缓冲或以其他方式加载视频以避免这种情况?
  3. 我已经看到了这个问题Android Streaming with MediaPlayer: Error(1, -1004) and 3GPP video,但答案并没有多大帮助。

    我找到了一个功能MediaPlayer.prepareAsync(),此处为https://developer.android.com/reference/android/media/MediaPlayer.html#prepareAsync(),当VideoView打开视频时会自动调用此功能,但这似乎无法正常工作

    修改

    因此,该解决方案将我带到了谷歌的ExoPlayer,这很容易换成我的VideoView,它就像一个魅力。

    1. ExoPlayer 添加为依赖
    2. 将布局中的视图更改为SimpleExoPlayerView
    3. 在活动
    4. 中初始化SimpleExoPlayer
    5. 初始化MediaSource并附加到播放器
    6. 不再需要时请记得release()
    7. 通过这种方式,流媒体可以无缝地工作。

4 个答案:

答案 0 :(得分:1)

Android MediaPlayer类不提供对较低级别设置的访问,例如缓冲区大小。

表单日志-1004表示: public static final int MEDIA_ERROR_IO

对我来说,这段代码很好用:

try{
    MediaController mediaController = new MediaController(this);

    Uri video = Uri.parse(url);

    mediaController.setAnchorView(videoView);
    videoView.requestFocus();
    videoView.setMediaController(mediaController);
    videoView.setVideoURI(video);

    videoView.setOnPreparedListener(new OnPreparedListener()
    {

        @Override
        public void onPrepared(MediaPlayer arg0)
        {
            videoView.start();
        }
    });
}catch (Exception e) {
    e.printStackTrace();
}

它似乎与流的来源有关,因为某些来源最终在设备上播放,但是我们需要的,从未播放过。

如果您可以看到,那么您就会通过日志知道您的信息流存在问题。

MediaHTTPConnection: readAt 26869519 / 241 => java.net.ProtocolException: unexpected end of stream

当预期的字节数(通常在响应的内容长度标头中设置)大于响应中的实际数据时,FixedLengthInputStream抛出该异常。检查内容长度标头是否正确。 (如果您要为内容长度提供自己的值,请确保它是正确的。)

有关详细信息,请参阅此帖unexpected end of stream error

其他这个也有帮助,请检查link如何为MediaPlayer创建长时间缓冲区

答案 1 :(得分:0)

Android上的VideoView并不是最好的,如果您使用VLC,您将获得更加流畅的缓存视频体验。

以下是有关如何从Android源代码编译VLC的说明:https://wiki.videolan.org/AndroidCompile/

答案 2 :(得分:0)

在流视频中使用VideoView可能会导致某些时候出现问题。我也面临很多问题。

使用JieCaoVideoPlayer库来传输Video。它允许更多选项。

  

https://github.com/lipangit/JieCaoVideoPlayer

只需导入

compile 'fm.jiecao:jiecaovideoplayer:5.3'

像普通视图一样添加布局

<fm.jiecao.jcvideoplayer_lib.JCVideoPlayerStandard
   android:id="@+id/videoplayer"
   android:layout_width="match_parent"
   android:layout_height="200dp"/>

设置您要播放的网址

JCVideoPlayerStandard jcVideoPlayerStandard = (JCVideoPlayerStandard) findViewById(R.id.videoplayer);
jcVideoPlayerStandard.setUp("http://2449.vod.myqcloud.com/2449_22ca37a6ea9011e5acaaf51d105342e3.f20.mp4"
                        , JCVideoPlayerStandard.SCREEN_LAYOUT_NORMAL, "Video");
jcVideoPlayerStandard.thumbImageView.setImage("http://p.qpic.cn/videoyun/0/2449_43b6f696980311e59ed467f22794e792_1/640");

答案 3 :(得分:0)

   implementation 'android.arch.lifecycle:extensions:1.1.0'
implementation 'android.arch.lifecycle:compiler:1.1.0'
implementation 'cn.jzvd:jiaozivideoplayer:7.0.5'

<cn.jzvd.JzvdStd
        android:id="@+id/video_player"
        android:layout_width="match_parent"
        android:layout_height="match_parent" />


    JzvdStd jzvdStd;

 jzvdStd = (JzvdStd) findViewById(R.id.video_player);

    jzvdStd.setUp(getIntent().getExtras().getString("id"), getIntent().getExtras().getString("title"), 0);

    jzvdStd.setScreenFullscreen();
    jzvdStd.backButton.setVisibility(View.GONE);
    jzvdStd.batteryLevel.setVisibility(View.GONE);
    jzvdStd.batteryTimeLayout.setVisibility(View.GONE);
    jzvdStd.titleTextView.setVisibility(View.GONE);
    jzvdStd.dismissBrightnessDialog();
    jzvdStd.dismissVolumeDialog();
    jzvdStd.dissmissControlView();
    jzvdStd.dismissProgressDialog();
    jzvdStd.startVideo();
    Glide.with(jzvdStd.getContext()).load(getIntent().getExtras().getString("thumb")).into(jzvdStd.thumbImageView);



 @Override
public void onBackPressed() {
    if (Jzvd.backPress()) {
        super.onBackPressed();
        return;
    }
    super.onBackPressed();

}

@Override
public void onDestroy() {
    super.onDestroy();

}

@Override
protected void onPause() {
    super.onPause();
    Jzvd.releaseAllVideos();
}