1、ijkplayer简介

ijkplayer是一个基于FFmpeg的轻量级Android/iOS视频播放器。FFmpeg的是全球领先的多媒体框架,能够解码,编码, 转码,复用,解复用,流,过滤器和播放大部分的视频格式。它提供了录制、转换以及流化音视频的完整解决方案。它包含了非常先进的音频/视频编解码库libavcodec,为了保证高可移植性和编解码质量,libavcodec里很多code都是从头开发的。

2、ijkplayer导入方式

ijk的导入方式有两种,第一种是使用gradle导入ijkplayer发布到jcenter,已经打包好的依赖包,第二种是去github中下载ijkplayer源码,自己进行编译。

  1. gradle方式导入:

    allprojects {
    repositories {
    jcenter()
    }
    }
    dependencies {
    # required, enough for most devices.
    compile 'tv.danmaku.ijk.media:ijkplayer-java:0.8.8'
    compile 'tv.danmaku.ijk.media:ijkplayer-armv7a:0.8.8' # Other ABIs: optional
    compile 'tv.danmaku.ijk.media:ijkplayer-armv5:0.8.8'
    compile 'tv.danmaku.ijk.media:ijkplayer-arm64:0.8.8'
    compile 'tv.danmaku.ijk.media:ijkplayer-x86:0.8.8'
    compile 'tv.danmaku.ijk.media:ijkplayer-x86_64:0.8.8' # ExoPlayer as IMediaPlayer: optional, experimental
    compile 'tv.danmaku.ijk.media:ijkplayer-exo:0.8.8'
    }
  2. 从github下载ijkplayer源码,进行编译

    1. 从github中拉取代码,并且cd到代码的目录下
      git clone https://github.com/Bilibili/ijkplayer.git ijkplayer-android
      cd ijkplayer-android
    2. 选择k0.8.8分支
      git checkout -B latest k0.8.8
    3. 执行ijkplayer项目中的init-android.sh进行初始化,包括了把ffmpeg的代码拉取到本地等操作
      ./init-android.sh
    4. 进入android/contrib目录下,clean,然后编译一下ffmpeg软解码库
      cd android/contrib
      ./compile-ffmpeg.sh clean
      ./compile-ffmpeg.sh all
    5. cd到上一级目录,也就是android目录下,运行compile-ijk.sh进行编译
      cd ..
      ./compile-ijk.sh all

    运行完成之后,会得到ijk所需的so文件,目录如图所示:



    在各个cpu架构下,都会有相应的so文件,比如armv7a的so文件,就在ijkplayer-armv7a/src/main/libs/armeabi-v7a目录下。

    编译完了so文件,我们看到上图中还有一个ijkplayer-java的文件,这个就是ijk中,jni调用so文件的包,我们可以编译这个包,也可以直接使用gradle方式,直接引入相应版本的ijkplayer-java包。

    编译ijkplayer的方式很简单,进入到ijkplayer-java工程下,运行代码:

    ./gradlew clean build

    然后把build/out中的aar,放到自己工程并依赖就可以了

    gradle依赖java包,与上面的gradle依赖一样,在自己工程下添加如下代码:

    由于我下载编译的是k0.8.8,所以我依赖的java包也是0.8.8,避免出现版本不兼容的问题

    compile 'tv.danmaku.ijk.media:ijkplayer-java:0.8.8'

    最后把自己编译的好so文件,放到工程的jniLib下面就好了。

3、简单的ijkplayer使用

  1. 引入.so文件:这里用的是源码编译后,导入so文件的方式,把相应的so文件放在自己工程的jniLibs文件夹中。
  2. 使用gradle导入ijkplayer-java包(此包是ijkplayer的jni代码)。
    compile 'tv.danmaku.ijk.media:ijkplayer-java:0.8.8'
  3. 自定义播放控件,类名自取,继承自FrameLayout。
    /**
    * 由ijkplayer提供,用于播放视频,需要给他传入一个surfaceView
    */
    private IMediaPlayer mMediaPlayer = null;
    /**
    * 视频文件地址
    */
    private String mPath ;
    /**
    * 视频请求header
    */
    private Map<String,String> mHeader; private SurfaceView mSurfaceView; private Context mContext;
    private boolean mEnableMediaCodec; public VideoPlayer(@NonNull Context context) {
    this(context, null);
    } public VideoPlayer(@NonNull Context context, @Nullable AttributeSet attrs) {
    this(context, attrs, 0);
    } public VideoPlayer(@NonNull Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
    super(context, attrs, defStyleAttr);
    init(context);
    } //初始化
    private void init(Context context) {
    mContext = context;
    setBackgroundColor(Color.BLACK);
    createSurfaceView();
    mAudioManager = (AudioManager)mContext.getApplicationContext().getSystemService(Context.AUDIO_SERVICE);
    mAudioFocusHelper = new AudioFocusHelper();
    } //创建surfaceView
    private void createSurfaceView() {
    mSurfaceView = new SurfaceView(mContext);
    mSurfaceView.getHolder().addCallback(new SurfaceHolder.Callback() {
    @Override
    public void surfaceCreated(SurfaceHolder surfaceHolder) { } @Override
    public void surfaceChanged(SurfaceHolder surfaceHolder, int i, int i1, int i2) {
    if (mMediaPlayer != null) {
    mMediaPlayer.setDisplay(surfaceHolder);
    }
    } @Override
    public void surfaceDestroyed(SurfaceHolder surfaceHolder) { }
    });
    LayoutParams layoutParams = new LayoutParams(LayoutParams.MATCH_PARENT
    , LayoutParams.MATCH_PARENT, Gravity.CENTER);
    // mSurfaceView.setLayoutParams(layoutParams);
    addView(mSurfaceView,0,layoutParams);
    } //创建一个新的player
    private IMediaPlayer createPlayer() {
    IjkMediaPlayer ijkMediaPlayer = new IjkMediaPlayer();
    ijkMediaPlayer.setOption(IjkMediaPlayer.OPT_CATEGORY_PLAYER, "opensles", 1); ijkMediaPlayer.setOption(IjkMediaPlayer.OPT_CATEGORY_PLAYER, "overlay-format", IjkMediaPlayer.SDL_FCC_RV32);
    ijkMediaPlayer.setOption(IjkMediaPlayer.OPT_CATEGORY_PLAYER, "framedrop", 1);
    ijkMediaPlayer.setOption(IjkMediaPlayer.OPT_CATEGORY_PLAYER, "start-on-prepared", 0); ijkMediaPlayer.setOption(IjkMediaPlayer.OPT_CATEGORY_PLAYER, "http-detect-range-support", 1); ijkMediaPlayer.setOption(IjkMediaPlayer.OPT_CATEGORY_CODEC, "skip_loop_filter", 48);
    ijkMediaPlayer.setOption(IjkMediaPlayer.OPT_CATEGORY_CODEC, "min-frames", 100);
    ijkMediaPlayer.setOption(IjkMediaPlayer.OPT_CATEGORY_PLAYER, "enable-accurate-seek", 1); ijkMediaPlayer.setVolume(1.0f, 1.0f); setEnableMediaCodec(ijkMediaPlayer,mEnableMediaCodec);
    return ijkMediaPlayer;
    } //设置是否开启硬解码
    private void setEnableMediaCodec(IjkMediaPlayer ijkMediaPlayer, boolean isEnable) {
    int value = isEnable ? 1 : 0;
    ijkMediaPlayer.setOption(IjkMediaPlayer.OPT_CATEGORY_PLAYER, "mediacodec", value);//开启硬解码
    ijkMediaPlayer.setOption(IjkMediaPlayer.OPT_CATEGORY_PLAYER, "mediacodec-auto-rotate", value);
    ijkMediaPlayer.setOption(IjkMediaPlayer.OPT_CATEGORY_PLAYER, "mediacodec-handle-resolution-change", value);
    } public void setEnableMediaCodec(boolean isEnable){
    mEnableMediaCodec = isEnable;
    } //设置ijkplayer的监听
    private void setListener(IMediaPlayer player){
    player.setOnPreparedListener(mPreparedListener);
    player.setOnVideoSizeChangedListener(mVideoSizeChangedListener);
    } /**
    * 设置自己的player回调
    */
    public void setVideoListener(VideoListener listener){
    mListener = listener;
    } //设置播放地址
    public void setPath(String path) {
    setPath(path,null);
    } public void setPath(String path,Map<String,String> header){
    mPath = path;
    mHeader = header;
    } //开始加载视频
    public void load() throws IOException {
    if(mMediaPlayer != null){
    mMediaPlayer.stop();
    mMediaPlayer.release();
    }
    mMediaPlayer = createPlayer();
    setListener(mMediaPlayer);
    mMediaPlayer.setDisplay(mSurfaceView.getHolder());
    mMediaPlayer.setDataSource(mContext, Uri.parse(mPath),mHeader); mMediaPlayer.prepareAsync();
    } public void start() {
    if (mMediaPlayer != null) {
    mMediaPlayer.start();
    mAudioFocusHelper.requestFocus();
    }
    } public void release() {
    if (mMediaPlayer != null) {
    mMediaPlayer.reset();
    mMediaPlayer.release();
    mMediaPlayer = null;
    mAudioFocusHelper.abandonFocus();
    }
    } public void pause() {
    if (mMediaPlayer != null) {
    mMediaPlayer.pause();
    mAudioFocusHelper.abandonFocus();
    }
    } public void stop() {
    if (mMediaPlayer != null) {
    mMediaPlayer.stop();
    mAudioFocusHelper.abandonFocus();
    }
    } public void reset() {
    if (mMediaPlayer != null) {
    mMediaPlayer.reset();
    mAudioFocusHelper.abandonFocus();
    }
    } public long getDuration() {
    if (mMediaPlayer != null) {
    return mMediaPlayer.getDuration();
    } else {
    return 0;
    }
    } public long getCurrentPosition() {
    if (mMediaPlayer != null) {
    return mMediaPlayer.getCurrentPosition();
    } else {
    return 0;
    }
    } public void seekTo(long l) {
    if (mMediaPlayer != null) {
    mMediaPlayer.seekTo(l);
    }
    } public boolean isPlaying(){
    if(mMediaPlayer != null) {
    return mMediaPlayer.isPlaying();
    }
    return false;
    }
  4. 之后在activity的布局中使用这个自定义videoView就可以了。

    activity里,基本代码:
    private VideoPlayer videoPlayer;
    
    @Override
    protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    videoPlayer = findViewById(R.id.video); videoPlayer.setVideoListener(this);
    videoPlayer.setPath("http://ipfs.ztgame.com.cn/QmRRGU4aUZEqJsHxKzBb1ns97GHw45eCRRZFe6Eu8GCmZ4.m3u8");
    try {
    videoPlayer.load();
    } catch (IOException e) {
    Toast.makeText(this,"播放失败",Toast.LENGTH_SHORT);
    e.printStackTrace();
    }
    } @Override
    public void onBufferingUpdate(IMediaPlayer iMediaPlayer, int i) { } @Override
    public void onCompletion(IMediaPlayer iMediaPlayer) { } @Override
    public boolean onError(IMediaPlayer iMediaPlayer, int i, int i1) {
    return false;
    } @Override
    public boolean onInfo(IMediaPlayer iMediaPlayer, int i, int i1) {
    return false;
    } @Override
    public void onPrepared(IMediaPlayer iMediaPlayer) {
    videoPlayer.start();
    } @Override
    public void onSeekComplete(IMediaPlayer iMediaPlayer) { } @Override
    public void onVideoSizeChanged(IMediaPlayer iMediaPlayer, int i, int i1, int i2, int i3) { }

4、结束语

到这里,播放器就可以运行了,如果有问题欢迎留言。

还有一个ijkplayer踩坑记录,里面会有一些我的踩坑记录,不定时更新。

最后附上播放器demo源码地址

demo代码也会不定时更新,慢慢变成一套可扩展,复用的播放器。

ijkplayer接入使用的更多相关文章

  1. 关于几个主流语音SDK的接入问题

    这两周都在忙着游戏上线还有接入游戏语音,两周分别接了腾讯语音和百度语音!!! 关于腾讯语音的一些问题 由于发现腾讯语音的在录完音频后的数据是编过码的所以出现了一些问题: *不能解码(腾讯方不提供解码算 ...

  2. Android 关于ijkplayer

    基于ijkplayer封装支持简单界面UI定制的视频播放器 可以解析ts格式的so库 怎样编译出可以解析ts等格式的so库?就是编译的时候需要在哪一步修改配置? 一些电视台的m3u8 CCTV1综合, ...

  3. 手机游戏渠道SDK接入工具项目分享(二)万事开头难

    一般接到任务后程序员们通常都开始着手进行技术调研了,但我这活是项目负责人.还有一大堆事情要先期准备,没人能帮忙. 一.人力配置 考虑的之前已经有一波人搞了大半年,但没有起色,先期也没有太大人力需求,所 ...

  4. 用java开发微信公众号:公众号接入和access_token管理(二)

    本文为原创,原始地址为http://www.cnblogs.com/fengzheng/p/5027630.html 上一篇说了微信开发的准备工作,准备工作完成之后,就要开始步入正题了.其实微信公众号 ...

  5. 微信SDK开发——接口接入

    园子里面很多关于微信接口开发的文章,Github也一堆的开源代码. 官方文档地址:http://mp.weixin.qq.com/wiki/home/index.html 接下来主要以代码为主,接口说 ...

  6. 【Android】 修复ijkPlayer进行m3u8 hls流播放时seek进度条拖动不准确的问题

    项目中使用的播放器是ijkPlayer,发现播放切片特点的hls流(m3u8格式的视频)拖动seekBar的时候会莫名的跳转或者seek不到准确的位置,发现网友也遇到了同样的问题,ijk的开发者也说明 ...

  7. 在基于MVC的Web项目中使用Web API和直接连接两种方式混合式接入

    在我之前介绍的混合式开发框架中,其界面是基于Winform的实现方式,后台使用Web API.WCF服务以及直接连接数据库的几种方式混合式接入,在Web项目中我们也可以采用这种方式实现混合式的接入方式 ...

  8. C#开发微信门户及应用(32)--微信支付接入和API封装使用

    在微信的应用上,微信支付是一个比较有用的部分,但也是比较复杂的技术要点,在微商大行其道的年代,自己的商店没有增加微信支付好像也说不过去,微信支付旨在为广大微信用户及商户提供更优质的支付服务,微信的支付 ...

  9. SDK接入(3)之iOS内支付(In-App Purchase)接入

    SDK接入(3)之iOS内支付(In-App Purchase)接入 继整理了Android平台的SDK接入过程.再来分享下iOS平台的内支付(In-App Purchase)接入,作为笔者在游戏开发 ...

随机推荐

  1. java实现定时任务解决方案

    在线corn表达式 1. 总结常见的实现定时任务的几种方法 thread实现 [原理:通过创建一个线程,让他在while循环里面一直运行,用sleep() 方法让其休眠从而达到定时任务的效果.] Ti ...

  2. Python_小程序(云开发)

    一.云开发API初始化 wx.cloud.init({ env:'test-x1dzi', //环境ID traceUser:true //是否在控制台查看用户信息 }) 二.云开发API初始化-服务 ...

  3. 图解算法——恢复一棵二叉搜索树(BST)

    题目来源 基础:给你二叉搜索树的根节点 root ,该树中的两个节点被错误地交换.请在不改变其结构的情况下,恢复这棵树. 进阶:使用 O(n) 空间复杂度的解法很容易实现.你能想出一个只使用常数空间的 ...

  4. Socket 编程简介

    Socket又称"套接字",应用程序通常通过"套接字"向网络发出请求或者应答网络请求,使主机间或者一台计算机上的进程间可以通讯. 本章节我们为大家接收 Perl ...

  5. 关于优先队列的总结II

    优先队列这个数据结构还是很有用的,可以帮我们解决很多棘手的排序的问题,所以再来细细看一下, priority_queue<Type, Container, Functional> Type ...

  6. Building an IMAP Email Client with PHP

    1 Building an IMAP Email Client with PHP http://www.toptal.com/php/building-an-imap-email-client-wit ...

  7. 前端安全 All In One

    前端安全 All In One refs xgqfrms 2012-2020 www.cnblogs.com 发布文章使用:只允许注册用户才可以访问!

  8. React + GraphQL 2020 速成课程

    React + GraphQL 2020 速成课程 technologies React (to build our user interface) GraphQL (to get and chang ...

  9. Web Performance API

    Web Performance API 性能监测/性能优化 https://developer.mozilla.org/en-US/docs/Web/API/Performance https://d ...

  10. React LifeCycle API

    React LifeCycle API old API & new API 不可以混用 demo https://codesandbox.io/s/react-parent-child-lif ...