在android系统中,有获取视频文件缩略图和时长api方法,系统的aip的方法最终是会调jni方法,这样使得我们在取一些不主流的视频格式文件的时候,获取不到缩略图和时长,然后即使是常见的编码,有时候也取不出来,对于某些机型来说。
一般系统自带的方法有两个如下:
一个是用MediaMetadataRetriever
MediaMetadataRetriever media = new MediaMetadataRetriever();
media.setDataSource(videoPath);
Bitmap bitmap = media.getFrameAtTime();
另外一个是contentprovider遍历cursor
Cursor thumbCursor = context.getApplicationContext().getContentResolver().query(
MediaStore.Video.Thumbnails.EXTERNAL_CONTENT_URI,
null, MediaStore.Video.Thumbnails.VIDEO_ID
+ "=" + id, null, null);
if (thumbCursor.moveToFirst()) {
albumPath = thumbCursor.getString(thumbCursor
.getColumnIndex(MediaStore.Video.Thumbnails.DATA));
Bitmap bitmap = BitmapFactory.decodeFile(albumPath);
}
//时长
cursor.getInt(cursor.getColumnIndexOrThrow(MediaStore.Video.Media.DURATION));
对于因为项目碰到一些一些非主流的视频编码格式的,上面两个方法都不能取到缩略图和时长,那么只能用地方三的编码库了。最常见莫过于ffmpeg,刚好github上面有这么一个项目。
https://github.com/wseemann/FFmpegMediaMetadataRetriever
只取了其中呢一个arm-v7 的arr包,效果很杠杠的,但是单arm-v7的包就已经4M大小了,对于关心安装包体积的来说,只取一个缩略图和时长,又不用ffmpeg进行音视频解码的视频编解码功能,是有点大材小用了。所以重新对该开源项目进行裁剪编译。
首先,本人是在linux环境下进行编译的,如果window 编译也可以的,之前我在window编译ffmpeg的,索性这次我就在linux下编译了。
然后项目工程中,配置好ndk路径,因为要读取local.properties的值的,然后需要关心的有两个文件,build-ffmpeg.sh,build_ffmpeg.sh。
在build-ffmpeg.sh中我把不需要的cpu类型给去掉
TARGET_ARMEABI_DIR=$WORKING_DIR/../jni/ffmpeg/ffmpeg/armeabi
TARGET_ARMEABIV7A_DIR=$WORKING_DIR/../jni/ffmpeg/ffmpeg/armeabi-v7a
TARGET_X86_DIR=$WORKING_DIR/../jni/ffmpeg/ffmpeg/x86
TARGET_MIPS_DIR=$WORKING_DIR/../jni/ffmpeg/ffmpeg/mips
TARGET_X86_64_DIR=$WORKING_DIR/../jni/ffmpeg/ffmpeg/x86_64
TARGET_ARMEABI_64_DIR=$WORKING_DIR/../jni/ffmpeg/ffmpeg/arm64-v8a
注释成
#TARGET_ARMEABI_DIR=$WORKING_DIR/../jni/ffmpeg/ffmpeg/armeabi
TARGET_ARMEABIV7A_DIR=$WORKING_DIR/../jni/ffmpeg/ffmpeg/armeabi-v7a
#TARGET_X86_DIR=$WORKING_DIR/../jni/ffmpeg/ffmpeg/x86
#TARGET_MIPS_DIR=$WORKING_DIR/../jni/ffmpeg/ffmpeg/mips
#TARGET_X86_64_DIR=$WORKING_DIR/../jni/ffmpeg/ffmpeg/x86_64
#TARGET_ARMEABI_64_DIR=$WORKING_DIR/../jni/ffmpeg/ffmpeg/arm64-v8a
然后把build_ffmpeg.sh中的路径配置好,比如说原项目是mac环境,
PREBUILT=NDK/toolchains/arm-linux-androideabi-4.9/prebuilt/linux-x86_64
最后是修改裁剪编译脚本了,其查看configure有哪些指令可以用,一搜一大堆。
原脚本内容:
pushd ffmpeg-$FFMPEG_VERSION
./configure --target-os=linux \
--incdir=$BUILD_DIR/$TARGET/include \
--libdir=$BUILD_DIR/$TARGET/lib \
--enable-cross-compile \
--extra-libs="-lgcc" \
--arch=$ARCH \
--cc=$PREBUILT/bin/$HOST-gcc \
--cross-prefix=$PREBUILT/bin/$HOST- \
--nm=$PREBUILT/bin/$HOST-nm \
--sysroot=$PLATFORM \
--extra-cflags="$OPTIMIZE_CFLAGS " \
--enable-shared \
--enable-small \
--extra-ldflags="-Wl,-rpath-link=$PLATFORM/usr/lib -L$PLATFORM/usr/lib -nostdlib -lc -lm -ldl -llog" \
--disable-ffplay \
--disable-ffmpeg \
--disable-ffprobe \
--disable-avfilter \
--disable-avdevice \
--disable-ffserver \
--disable-doc \
--disable-avdevice \
--disable-swresample \
--disable-postproc \
--disable-avfilter \
--disable-gpl \
--disable-encoders \
--disable-hwaccels \
--disable-muxers \
--disable-bsfs \
--disable-protocols \
--disable-indevs \
--disable-outdevs \
--disable-devices \
--disable-filters \
--enable-encoder=png \ 编码格式开了png,因为要什么图片,其他编码都是禁用了--disable-encoders \
--enable-protocol=file,http,https,mmsh,mmst,pipe,rtmp,rtmps,rtmpt,rtmpts,rtp \
--disable-debug \
--disable-asm \
$ADDITIONAL_CONFIGURE_FLAG
然后我们的做法是把编码(encoder)禁用,封装(muxers)禁用,然后开放常用普遍的解码(decoders),解封装(demuxers)。如果后续项目发现有些编码格式取不出来,再把此编码格式放进去重新编译包。
最后多加了一下两句:
--disable-decoders --enable-decoder=vp9 --enable-decoder=h264 --enable-decoder=mpeg4 --enable-decoder=aac \
--disable-demuxers --enable-demuxer=rtsp --enable-demuxer=rtp --enable-demuxer=flv --enable-demuxer=h264 \
然后我编译出来封装成arr,就只有1.2M了,大体过程就是这样了
附上编译包地址:
https://github.com/liangzs/ffmpeg/blob/master/armv7-fmmr.aar
网友评论