基本思路:先编译出ffmpeg多个静态库库,然后使用工具将其打包成一个ffmpeg.so库
编译环境
macOS High Sierri 10.13.3
android-ndk-r14b
FFmpeg 3.4.2
操作步骤
1. 直接运行下面的build_android.sh
脚本即可:
#!/bin/bash
PLATFORM=/Users/lake/test/android-ndk-r14b/platforms/android-19/arch-arm/
TOOLCHAIN=/Users/lake/test/android-ndk-r14b/toolchains/arm-linux-androideabi-4.9/prebuilt/darwin-x86_64
PREFIX=./android
function build_one
{
./configure \
--prefix=$PREFIX \
--target-os=android \
--cross-prefix=$TOOLCHAIN/bin/arm-linux-androideabi- \
--arch=arm \
--sysroot=$PLATFORM \
--extra-cflags="-I$PLATFORM/usr/include" \
--cc=$TOOLCHAIN/bin/arm-linux-androideabi-gcc \
--nm=$TOOLCHAIN/bin/arm-linux-androideabi-nm \
--disable-shared \
--disable-ffmpeg \
--disable-ffplay \
--disable-ffprobe \
--disable-ffserver \
--disable-doc \
--disable-symver \
--enable-small \
--enable-gpl \
--enable-asm \
--enable-jni \
--enable-mediacodec \
--enable-decoder=h264_mediacodec \
--enable-hwaccel=h264_mediacodec \
--enable-decoder=hevc_mediacodec \
--enable-decoder=mpeg4_mediacodec \
--enable-decoder=vp8_mediacodec \
--enable-decoder=vp9_mediacodec \
--enable-nonfree \
--enable-version3 \
--extra-cflags="-Os -fpic $ADDI_CFLAGS" \
--extra-ldflags="$ADDI_LDFLAGS" \
$ADDITIONAL_CONFIGURE_FLAG
make clean
make j8
make install
$TOOLCHAIN/bin/arm-linux-androideabi-ld \
-rpath-link=$PLATFORM/usr/lib \
-L$PLATFORM/usr/lib \
-L$PREFIX/lib \
-soname libffmpeg.so -shared -nostdlib -Bsymbolic --whole-archive --no-undefined -o \
$PREFIX/libffmpeg.so \
libavcodec/libavcodec.a \
libavfilter/libavfilter.a \
libswresample/libswresample.a \
libavformat/libavformat.a \
libavutil/libavutil.a \
libswscale/libswscale.a \
libavdevice/libavdevice.a \
libpostproc/libpostproc.a \
-lc -lm -lz -ldl -llog --dynamic-linker=/system/bin/linker \
$TOOLCHAIN/lib/gcc/arm-linux-androideabi/4.9.x/libgcc.a
}
# arm v7vfp
CPU=arm
OPTIMIZE_CFLAGS="-mfloat-abi=softfp -mfpu=vfp -marm -march=$CPU "
ADDI_CFLAGS="-marm"
build_one
说明:Configure的时候设置为编译静态库
–enable-static –disable-shared
。
编译完成后生成的是:
libavcodec.a
libavfilter.a
libswresample.a
libavformat.a
libavutil.a
libswscale.a
该脚本运行完后,会把上述的*.a文件打包为一个so文件:
libffmpeg.so
合并后的类库使用起来和合并前的类库使用方法没有区别。
2. 单个库版FFmpeg.so使用方法
使用CMake编译
目前越来越多的人使用cmake来进行工程编译了,下面是使用cmake的配置代码:
#CMake版本信息
cmake_minimum_required(VERSION 3.4.1)
#ffmpeg so文件路径
set(lib_src_DIR ${CMAKE_SOURCE_DIR}/src/main/jniLibs/${ANDROID_ABI})
#配置加载native依赖
include_directories(${CMAKE_SOURCE_DIR}/src/main/cpp/include)
#动态方式加载ffmepg的so文件 第三方库加载方式
add_library(ffmpeg SHARED IMPORTED)
#引入libffmpeg.so文件
set_target_properties(ffmpeg PROPERTIES IMPORTED_LOCATION${lib_src_DIR}/libffmpeg.so)
#CPP文件夹下待编译的c文件
add_library(native-lib SHARED ${CMAKE_SOURCE_DIR}/src/main/cpp/native-lib.cpp)
#C 日志 ndk官方库find_library(log-lib log)
#静态库与动态库进行链接 相当于gcc命令行参数 -l。
target_link_libraries(native-lib
android
ffmpeg
${log-lib})
MainActivity.java 加载SO库的代码
static {
System.loadLibrary("ffmpeg");
System.loadLibrary("native-lib");
}
使用Android.mk编译
如果使用老式的Android.mk
文件,则将下面的代码添加到你的Android.mk
即可。
LOCAL_PATH := $(call my-dir)
# FFmpeg library
include $(CLEAR_VARS)
LOCAL_MODULE := ffmpeg
LOCAL_SRC_FILES := libffmpeg.so
include $(PREBUILT_SHARED_LIBRARY)
# Program
include $(CLEAR_VARS)
LOCAL_MODULE := ffmpegdemo
LOCAL_SRC_FILES := ffmpegdemo.c
LOCAL_C_INCLUDES += $(LOCAL_PATH)/include
LOCAL_LDLIBS := -llog -lz
LOCAL_SHARED_LIBRARIES := ffmpeg
include $(BUILD_SHARED_LIBRARY)
MainActivity.java 加载SO库的代码
static {
System.loadLibrary("ffmpeg");
System.loadLibrary("ffmpegdemo");
}