Warning: file_get_contents(/data/phpspider/zhask/data//catemap/3/android/195.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
Android 自适应hls流媒体播放器_Android_Http Live Streaming_Exoplayer_Adaptive Bitrate - Fatal编程技术网

Android 自适应hls流媒体播放器

Android 自适应hls流媒体播放器,android,http-live-streaming,exoplayer,adaptive-bitrate,Android,Http Live Streaming,Exoplayer,Adaptive Bitrate,我正在寻找如何为HLS Adaptive流媒体实现ExoPlayer的简单好例子/解释。我是一个新手,没有经验和知识,所以我可以从git上的代码示例中找出如何做到这一点 有太多的“活动部件”,所以初学者可以理解并在自己的项目中重用它 有人能帮助我学习和理解如何使用/实现ExoPlayer,以实现此功能吗 谢谢 开始使用ExoPlayer最简单的方法是将其添加为渐变依赖项。您需要确保项目根目录中的build.gradle文件中包含jcenter存储库: 存储库{ jcenter() } 接下来,在

我正在寻找如何为
HLS Adaptive
流媒体实现
ExoPlayer
的简单好例子/解释。我是一个新手,没有经验和知识,所以我可以从git上的代码示例中找出如何做到这一点

有太多的“活动部件”,所以初学者可以理解并在自己的项目中重用它

有人能帮助我学习和理解如何使用/实现
ExoPlayer
,以实现此功能吗


谢谢

开始使用ExoPlayer最简单的方法是将其添加为渐变依赖项。您需要确保项目根目录中的build.gradle文件中包含jcenter存储库:

存储库{
jcenter()
}

接下来,在模块的build.gradle文件中包括以下内容:

编译'com.google.android.exoplayer:exoplayer:r2.2.0'

1。您的布局文件

    <?xml version="1.0" encoding="utf-8"?>
<FrameLayout
    android:id="@+id/activity_main"
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <com.google.android.exoplayer2.ui.SimpleExoPlayerView
        android:id="@+id/player_view"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:focusable="true"
        app:resize_mode="fill"/>

    <ProgressBar
        android:id="@+id/progressBar"
        style="?android:attr/progressBarStyle"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="center"
        android:visibility="gone"/>

</FrameLayout>
我想这对初学者来说已经足够了。还要记住,该库的标准音频和视频组件依赖于Android的MediaCodec API,该API是在Android 4.1(API级别16)中发布的。因此,它无法在android 4.0及以下版本上运行

不要忘记将此权限添加到
清单文件中


来自@Vicky的答案将起作用,但有一个缺陷

传递给轨迹选择器的带宽计必须与数据源工厂使用的带宽计相同。数据源工厂通过调用BW meter方法来维护带宽估计值,自适应航迹选择过程获得估计值以决定要适应哪个航迹

如果它们不是同一个实例,自适应选择总是得到-1作为BW,并选择一些中间地带选项


ExoPlayer演示应用程序也存在此缺陷。它们将false传递给buildDataSource()中的useBwMeter,这意味着不更新BW估算编辑:实际上,此BW度量是用于清单加载程序的。它不需要使用BW测量仪。

谢谢你,Vicky!请告知exoplayer是否会根据带宽自动选择视频质量?我看到了必须实现的方法,但我不明白如何使用它?请跟我谈谈。谢谢:)Exoplayer将自动选择带宽上的视频。您不必担心这个问题。您是否为此代码获得了
BehindLiveWindowException
?我看到在上有一个关于这个问题的bug。我已经从
ExoPlayer.EventListener
接口中重写了方法
onplayerror
,但运气不好。有什么建议吗?当然,这是我实现的viewModel类,我认为这是一个bug,在github上,他们告诉你重新初始化mediasource,你已经这样做了。有没有办法通过exo player实现RTMP自适应流媒体?
    public class VideoPlayerActivity extends AppCompatActivity implements ExoPlayer.EventListener {

    private SimpleExoPlayerView simpleExoPlayerView;
    private String hlsVideoUri = "http://playertest.longtailvideo.com/adaptive/bbbfull/bbbfull.m3u8";
    private SimpleExoPlayer player;
    private ProgressBar progressBar;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_video_player);

        // 1. Create a default TrackSelector
        Handler mainHandler = new Handler();
        BandwidthMeter bandwidthMeter = new DefaultBandwidthMeter();
        TrackSelection.Factory videoTrackSelectionFactory = new AdaptiveVideoTrackSelection.Factory(bandwidthMeter);
        TrackSelector trackSelector = new DefaultTrackSelector(videoTrackSelectionFactory);

        // 2. Create a default LoadControl
        LoadControl loadControl = new DefaultLoadControl();


        // 3. Create the player
        player = ExoPlayerFactory.newSimpleInstance(this, trackSelector, loadControl);

        simpleExoPlayerView = (SimpleExoPlayerView) findViewById(R.id.player_view);
        simpleExoPlayerView.setPlayer(player);

        // Measures bandwidth during playback. Can be null if not required.
        DefaultBandwidthMeter defaultBandwidthMeter = new DefaultBandwidthMeter();
        // Produces DataSource instances through which media data is loaded.
        DataSource.Factory dataSourceFactory = new DefaultDataSourceFactory(this,
                Util.getUserAgent(this, "Exo2"), defaultBandwidthMeter);
        // Produces Extractor instances for parsing the media data.
        ExtractorsFactory extractorsFactory = new DefaultExtractorsFactory();
        // This is the MediaSource representing the media to be played.
        HlsMediaSource hlsMediaSource = new HlsMediaSource(Uri.parse(hlsVideoUri), dataSourceFactory, mainHandler, new AdaptiveMediaSourceEventListener() {
            @Override
            public void onLoadStarted(DataSpec dataSpec, int dataType, int trackType, Format trackFormat, int trackSelectionReason, Object trackSelectionData, long mediaStartTimeMs, long mediaEndTimeMs, long elapsedRealtimeMs) {

            }

            @Override
            public void onLoadCompleted(DataSpec dataSpec, int dataType, int trackType, Format trackFormat, int trackSelectionReason, Object trackSelectionData, long mediaStartTimeMs, long mediaEndTimeMs, long elapsedRealtimeMs, long loadDurationMs, long bytesLoaded) {

            }

            @Override
            public void onLoadCanceled(DataSpec dataSpec, int dataType, int trackType, Format trackFormat, int trackSelectionReason, Object trackSelectionData, long mediaStartTimeMs, long mediaEndTimeMs, long elapsedRealtimeMs, long loadDurationMs, long bytesLoaded) {

            }

            @Override
            public void onLoadError(DataSpec dataSpec, int dataType, int trackType, Format trackFormat, int trackSelectionReason, Object trackSelectionData, long mediaStartTimeMs, long mediaEndTimeMs, long elapsedRealtimeMs, long loadDurationMs, long bytesLoaded, IOException error, boolean wasCanceled) {

            }

            @Override
            public void onUpstreamDiscarded(int trackType, long mediaStartTimeMs, long mediaEndTimeMs) {

            }

            @Override
            public void onDownstreamFormatChanged(int trackType, Format trackFormat, int trackSelectionReason, Object trackSelectionData, long mediaTimeMs) {

            }
        });

        player.addListener(this);
        player.prepare(hlsMediaSource);
        simpleExoPlayerView.requestFocus();
        player.setPlayWhenReady(true);

        progressBar = (ProgressBar) findViewById(R.id.progressBar);
    }

    @Override
    public void onTimelineChanged(Timeline timeline, Object manifest) {

    }

    @Override
    public void onTracksChanged(TrackGroupArray trackGroups, TrackSelectionArray trackSelections) {

    }

    @Override
    public void onLoadingChanged(boolean isLoading) {

    }

    @Override
    public void onPlayerStateChanged(boolean playWhenReady, int playbackState) {

        switch (playbackState) {
            case Player.STATE_BUFFERING:
                //You can use progress dialog to show user that video is preparing or buffering so please wait
                progressBar.setVisibility(View.VISIBLE);
                break;
            case Player.STATE_IDLE:
                //idle state
                break;
            case Player.STATE_READY:
                // dismiss your dialog here because our video is ready to play now
                progressBar.setVisibility(View.GONE);
                break;
            case Player.STATE_ENDED:
                // do your processing after ending of video
                break;
        }
    }

    @Override
    public void onPlayerError(ExoPlaybackException error) {

        AlertDialog.Builder adb = new AlertDialog.Builder(VideoPlayerActivity.this);
        adb.setTitle("Could not able to stream video");
        adb.setMessage("It seems that something is going wrong.\nPlease try again.");
        adb.setPositiveButton("OK", new DialogInterface.OnClickListener() {
            @Override
            public void onClick(DialogInterface dialog, int which) {
                dialog.dismiss();
                finish(); // take out user from this activity. you can skip this
            }
        });
        AlertDialog ad = adb.create();
        ad.show();
    }

    @Override
    public void onPositionDiscontinuity() {

    }

    @Override
    protected void onPause() {
        super.onPause();
        if (player != null) {
            player.setPlayWhenReady(false); //to pause a video because now our video player is not in focus
        }
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        player.release();
    }
}