ijk
ijkplayer的github地址
ijkplayer的编译
gradle
现在可以直接使用gradle引入ijkplayer了。但是,如果需要对更多格式进行支持,还是需要自己编译一次。
allprojects {
repositories {
jcenter()
}
}
dependencies {
# required, enough for most devices.
compile 'tv.danmaku.ijk.media:ijkplayer-java:0.8.1.2'
compile 'tv.danmaku.ijk.media:ijkplayer-armv7a:0.8.1.2'
# Other ABIs: optional
compile 'tv.danmaku.ijk.media:ijkplayer-armv5:0.8.1.2'
compile 'tv.danmaku.ijk.media:ijkplayer-arm64:0.8.1.2'
compile 'tv.danmaku.ijk.media:ijkplayer-x86:0.8.1.2'
compile 'tv.danmaku.ijk.media:ijkplayer-x86_64:0.8.1.2'
# ExoPlayer as Imediaplayer: optional, experimental
compile 'tv.danmaku.ijk.media:ijkplayer-exo:0.8.1.2'
}
编译
如果我们需要获取更多的视频格式支持(比如mkv,rmvb等),需要自己进行编译。我这里有个成品,包括一个小的demo,不想编译的同学可以自提。github地址。
我是在ubuntu下编译的。具体的方法官方的GitHub有,我总结一下,基本就是对着终端输入指令:
自行在ubuntu下配置好Android的sdk和ndk。
安装git和yasm。打开终端,依次输入如下指令:
sudo apt-get update
sudo apt-get install git
sudo apt-get install yasm
- 从github中拉取代码,并且cd到代码的目录下
git clone https://github.com/Bilibili/ijkplayer.git ijkplayer-android
cd ijkplayer-android
- 把代码更新到最新的版本。最新的版本号可以看GitHub。输入指令:
git checkout -B latest k0.8.1.2
- 初始化,包括了把ffmpeg的代码拉取到本地等操作,输入指令:
./init-android.sh
- clean一下,输入指令:
cd android/contrib
./compile-ffmpeg.sh clean
- 编译ffmpeg软解码库,输入指令:
./compile-ffmpeg.sh all
- cd到上一级目录,输入指令:
cd ..
- 得到ijkplayer的项目,输入指令:
./compile-ijk.sh all
其实就是跟着步骤在命令行中敲指令。看清楚指令,不要敲错,应该很快就可以得到我们ijkplayer的项目了。这个项目已经可以支持所有的视频格式了。
在这个项目中,有好多好多个module,有exmaple,ijkplayer-java,ijkplayer-armv5等。分别是例子,ijkplayer java层的代码,适配不同cpu的native层代码等。
简单的ijkplayer使用
下面是我使用的方法。可以直接看这个demo:github地址。
- 引入依赖。
我使用的ijkplayer的支持库,是编译得来的。所以我依赖了这些module
compile project(':ijkplayer-java')
compile project(':ijkplayer-armv5')
compile project(':ijkplayer-armv7a')
compile project(':ijkplayer-arm64')
compile project(':ijkplayer-x86')
compile project(':ijkplayer-x86_64')
- 自定义播放控件
ijkplayer提供的没有提供一个播放器控件给我们使用,所以我们自己定义一个。主要是创建一个surfaceview,把它赋给IMediaPlayer。
/**
* 由ijkplayer提供,用于播放视频,需要给他传入一个surfaceView
*/
private IMediaPlayer mMediaPlayer = null;
/**
* 视频文件地址
*/
private String mPath = "";
private SurfaceView surfaceView;
private VideoPlayerListener listener;
private context mContext;
public VideoPlayerIJK(@NonNull Context context) {
super(context);
initVideoView(context);
}
public VideoPlayerIJK(@NonNull Context context, @Nullable AttributeSet attrs) {
super(context, attrs);
initVideoView(context);
}
public VideoPlayerIJK(@NonNull Context context, @Nullable AttributeSet attrs, @AttrRes int defStyleAttr) {
super(context, attrs, defStyleAttr);
initVideoView(context);
}
private void initVideoView(Context context) {
mContext = context;
//获取焦点,不知道有没有必要~。~
setFocusable(true);
}
/**
* 设置视频地址。
* 根据是否第一次播放视频,做不同的操作。
*
* @param path the path of the video.
*/
public void setVideoPath(String path) {
if (TextUtils.equals("", mPath)) {
//如果是第一次播放视频,那就创建一个新的surfaceView
mPath = path;
createSurfaceView();
} else {
//否则就直接load
mPath = path;
load();
}
}
/**
* 新建一个surfaceview
*/
private void createSurfaceView() {
//生成一个新的surface view
surfaceView = new SurfaceView(mContext);
surfaceView.getHolder().addCallback(new LmnSurfaceCallback());
layoutparams layoutParams = new LayoutParams(LayoutParams.MATCH_parent
, LayoutParams.MATCH_PARENT, Gravity.CENTER);
surfaceView.setLayoutParams(layoutParams);
this.addView(surfaceView);
}
/**
* surfaceView的监听器
*/
private class LmnSurfaceCallback implements surfaceholder.Callback {
@Override
public void surfaceCreated(SurfaceHolder holder) {
}
@Override
public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {
//surfaceview创建成功后,加载视频
load();
}
@Override
public void surfacedestroyed(SurfaceHolder holder) {
}
}
/**
* 加载视频
*/
private void load() {
//每次都要重新创建IMediaPlayer
createPlayer();
try {
mMediaPlayer.setdatasource(mPath);
} catch (IOException e) {
e.printstacktrace();
}
//给mediaPlayer设置视图
mMediaPlayer.setdisplay(surfaceView.getHolder());
mMediaPlayer.prepareAsync();
}
/**
* 创建一个新的player
*/
private void createPlayer() {
if (mMediaPlayer != null) {
mMediaPlayer.stop();
mMediaPlayer.setDisplay(null);
mMediaPlayer.release();
}
IjkMediaPlayer ijkMediaPlayer = new IjkMediaPlayer();
ijkMediaPlayer.native_setLogLevel(IjkMediaPlayer.IJK_LOG_DEBUG);
//开启硬解码 ijkMediaPlayer.setOption(IjkMediaPlayer.OPT_CATEGORY_PLAYER, "mediacodec", 1);
mMediaPlayer = ijkMediaPlayer;
if (listener != null) {
mMediaPlayer.setOnPreparedListener(listener);
mMediaPlayer.setOnInfoListener(listener);
mMediaPlayer.setOnSeekCompleteListener(listener);
mMediaPlayer.setOnBufferingUpdateListener(listener);
mMediaPlayer.setOnERRORListener(listener);
}
}
public void setListener(VideoPlayerListener listener) {
this.listener = listener;
if (mMediaPlayer != null) {
mMediaPlayer.setOnPreparedListener(listener);
}
}
/**
* -------======--------- 下面封装了一下控制视频的方法
*/
public void start() {
if (mMediaPlayer != null) {
mMediaPlayer.start();
}
}
public void release() {
if (mMediaPlayer != null) {
mMediaPlayer.reset();
mMediaPlayer.release();
mMediaPlayer = null;
}
}
public void pause() {
if (mMediaPlayer != null) {
mMediaPlayer.pause();
}
}
public void stop() {
if (mMediaPlayer != null) {
mMediaPlayer.stop();
}
}
public void reset() {
if (mMediaPlayer != null) {
mMediaPlayer.reset();
}
}
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);
}
}
}
我们这个控件继承自framelayout。这个控件负责存放一个surfaceView和一个IMediaPlayer 。
3,设置监听器,我自己定义了一个监听器,继承了IMediaPlayer的n个listener。
public abstract class VideoPlayerListener implements IMediaPlayer.OnBufferingUpdateListener, IMediaPlayer.OnCompletionListener, IMediaPlayer.OnPreparedListener, IMediaPlayer.OnInfoListener, IMediaPlayer.OnVideoSizeChangedListener, IMediaPlayer.OnErrorListener, IMediaPlayer.OnSeekCompleteListener {
}
当我们继承这个抽象类的时候,最重要的是在onPrepared()方法中,让视频开始播放:
ijkPlayer.setListener(new VideoPlayerListener() {
@Override
public void onBufferingUpdate(IMediaPlayer mp, int percent) {
}
@Override
public void onCompletion(IMediaPlayer mp) {
}
@Override
public boolean onError(IMediaPlayer mp, int what, int extra) {
return false;
}
@Override
public boolean onInfo(IMediaPlayer mp, int what, int extra) {
return false;
}
@Override
public void onPrepared(IMediaPlayer mp) {
// 视频准备好播放了,但是他不会自动播放,需要手动让他开始。
mp.start();
}
@Override
public void onSeekComplete(IMediaPlayer mp) {
}
@Override
public void onVideoSizeChanged(IMediaPlayer mp, int width, int height, int sar_num, int sar_den) {
//在此可以获取到视频的宽和高
}
});
- 在xml中放入播放器控件
- 在activity中加载so包,设置监听器,设置路径
//加载native库
try {
IjkMediaPlayer.loadLibrariesOnce(null);
IjkMediaPlayer.native_profileBegin("libijkplayer.so");
} catch (Exception e) {
this.finish();
}
ijkPlayer.setListener(...)
ijkPlayer.setVideoPath(path);
记得在onStop()方法中关闭native库
IjkMediaPlayer.native_profileEnd();
这样就完成了一个简单的视频播放器了。
注意事项
1,IjkMediaPlayer的setDataSource是不建议重新赋值的,每次更改视频源都需要:player.release() -> create new player -> player.setDataSource。
但是SurfaceView并不需要每次都重新创建。
2,设置倍速播放:IjkMediaPlayer.setSpeed();
3,一些重要的视频信息返回码(这些信息返回码可以从监听器的onInfo()方法中获得):
int MEDIA_INFO_VIDEO_RENDERING_START = 3;//视频准备渲染
int MEDIA_INFO_BUFFERING_START = 701;//开始缓冲
int MEDIA_INFO_BUFFERING_END = 702;//缓冲结束
int MEDIA_INFO_VIDEO_ROTATION_CHANGED = 10001;//视频选择信息
int MEDIA_ERROR_SERVER_DIED = 100;//视频中断,一般是视频源异常或者不支持的视频类型。
int MEDIA_ERROR_IJK_PLAYER = -10000,//一般是视频源有问题或者数据格式不支持,比如音频不是AAC之类的
int MEDIA_ERROR_NOT_valid_FOR_progressIVE_PLAYBACK = 200;//数据错误没有有效的回收
参考资料:
在ubuntu下编译ijkplayer-android
视频直播技术(四):使用Ijkplayer播放直播视频
1、Ijkplayer 编码
IjkPlayer支持硬解码和软解码。 软解码时不会旋转视频角度这时需要你通过onInfo
的what == IMediaPlayer.MEDIA_INFO_VIDEO_ROTATION_CHANGED
去获取角度,自己旋转画面。或者开启硬解硬解码,不过硬解码容易造成黑屏无声(硬件兼容问题),下面是设置硬解码相关的代码
mediaPlayer.setOption(IjkMediaPlayer.OPT_CATEGORY_PLAYER, "mediacodec", 1); mediaPlayer.setOption(IjkMediaPlayer.OPT_CATEGORY_PLAYER, "mediacodec-auto-rotate", 1); mediaPlayer.setOption(IjkMediaPlayer.OPT_CATEGORY_PLAYER, "mediacodec-handle-resolution-change", 1);
2、Ijkplayer倍速播放
目前IjkPlayer最新版本支持倍速播放(版本号0.7.7.1)。在早期版本对倍速播放支持的不是很好,只有6.0及以上的手机才能进行倍速播放。但是目前虽然所有的机型都支持倍速,但是6.0以下的手机明显有声调变化,这样用户体验不是很好(针对变调的文章会在以后更新)。下面是最新的倍速设置代码:
public void setSpeed(float speed) { _setPropertyFloat(FFP_PROP_FLOAT_PLAYBACK_RATE, speed); }
老版本代码:
@TargetApi(build.VERSION_CODES.M) public void setSpeed(float speed) { _setPropertyFloat(FFP_PROP_FLOAT_PLAYBACK_RATE, speed); }
老版本代码只能设置速度,不能获取,所以基本上不能用(而且机型方面只支持6.0以上的机型)。
3、Ijkplayer的一些重要的视频返回码
int MEDIA_INFO_VIDEO_RENDERING_START = 3;//视频准备渲染 int MEDIA_INFO_BUFFERING_START = 701;//开始缓冲 int MEDIA_INFO_BUFFERING_END = 702;//缓冲结束 int MEDIA_INFO_VIDEO_ROTATION_CHANGED = 10001;//视频选择信息 int MEDIA_ERROR_SERVER_DIED = 100;//视频中断,一般是视频源异常或者不支持的视频类型。 int MEDIA_ERROR_IJK_PLAYER = -10000,//一般是视频源有问题或者数据格式不支持,比如音频不是AAC之类的 int MEDIA_ERROR_NOT_VALID_FOR_PROGRESSIVE_PLAYBACK = 200;//数据错误没有有效的回收
4、Ijkplayer调用seekTo存在的问题
部分视频播放时,调用seekTo的时候,会跳回到拖动前的位置,这是因为视频的关键帧的问题(GOP导致的),视频压缩比较高,而seek只支持关键帧,出现这个情况就是原始的视频文件中i帧比较少,播放器会在拖动的位置找最近的关键帧。所以,目前针对此问题IjkPlayer无解。
5. Ijkplayer音视频不同步问题
在开始使用过程当中对rtmp视频流进行播放,会出现严重的视频音频不同步现象,并且随着播放的时间越长,视频与音频的差距越大。具体原因是CPU在处理视频帧的时候处理得太慢,默认的音视频同步方案是视频同步到音频, 导致了音频播放过快,视频跟不上。
{ "framedrop", "drop frames when cpu is too slow", OPTION_offset(framedrop), OPTION_INT(0, -1, 120) },
framedrop 控制着允许丢帧的范围。可以通过修改 framedrop 的数值来解决不同步的问题,framedrop 是在视频帧处理不过来的时候丢弃一些帧达到同步的效果。具体设置:
player.setOption(IjkMediaPlayer.OPT_CATEGORY_PLAYER, "framedrop", 5);
framedrop 的具体大小根据实际情况而定, 一般丢太多帧也不好,会影响用户的观看体验。
6、如何支持https链接播放?
如果你的项目要进行加密播放HLS协议的视频,要想支持https,须要在普通编译的基础上,进行一些配置。
接下来我们来编译openssl
a). init openssl
$ cd .. 进入到ijkplayer的目下 $ ./init-android-openssl.sh 去远程仓库拉取openssl的远程代码,如果是iOS的,这里是init-ios-openssl.h
b). compile openssl
$ cd android/contrib $ ./compile-openssl.sh clean $ ./compile-openssl.sh all
经过以上步骤已经编译好openssl了,然后我们执行一下方法
$./compile-ffmpeg.sh clean 编译ffmpeg软解码库,这个过程会生成各种架构的ffmpeg 这个过程比较耗时 $./compile-ffmpeg.sh all
7、Ijkplayer使用小技巧
a. 下载速度可以通过IjkMediaPlayer的 getTcpSpeed
获取。
b. 高分辨率开启硬解码,不支持的话会自动切换到软解,就算开启mediacodec,如果设备不支持,显示的解码器也是avcodec软解。
c. IjkMediaPlayer.setOption可配置的对应头文件参考:ff_ffplay_options。
d. 设置cookie 可以通过ijkPlayer的public void setDataSource(String path, Map<String, String> headers) 的header实现设置,参考ijkPlayer的issues-1150,headers也是在内部被转化为何issuses一样的setOption方法
Android_观看视频卡顿原因及解决办法
在LiveView界面观看视频时,有时会出现画面卡顿的情况,有一种情况可能是只解了I帧的原因。在调用startShow(int avChannel, boolean clearBuf,
boolean runSoftwareDecode, boolean isDropFrame)方法时将isDropFrame设置成了true。
解决方法:将isDropFrame设置为false即可。
相关阅读
Android Studio中app出现红叉的解决方案 如果你在网上搜索了一大圈之后发现还是没有解决你的问题,或许这篇文章能帮到你,同事的开
【知识整理】安卓(Android)应用上传360应用市场步骤详
一。将开发好的安卓应用打成apk包。二。将应用上传至360应用市场1.注册360账号:网址:http://dev.360.cn/2.点击“提交软件”按钮3.
android 8.0 MountService -->StorageManagerService
android 8.0 MountService -->StorageManagerService从android开始一直以来的MountService变更成了StorageManagerService。位置:
面试的时候总会遇到一些各种各样的面试题,而且这些面试题很多都是关于平时容易疏忽的理论方面的,所以整理一份Android高级开发工
android控件的属性 android控件的属性本节描述android空间的位置,内容等相关属性及属性的含义 第一类:属性值为true或false androi