音视频学习:使用NDK编译FFmpeg动态库

news2025/5/14 15:10:13

1. 环境

1.1 基础配置

  • NDK 22b (r22b)
  • FFmpeg 4.4
  • Ubuntu 22.04

1.2 下载ffmpeg

官网提供了 .tar.xz 包,可以直接下载解压:

wget https://ffmpeg.org/releases/ffmpeg-4.4.tar.xz
tar -xvf ffmpeg-4.4.tar.xz
cd ffmpeg-4.4

1.3 安装基础工具链

sudo apt-get update && sudo apt-get install \
build-essential autoconf automake libtool \
pkg-config cmake git wget unzip yasm

可能不包含全部,遇到报错缺少的工具链的,把报错抛给AI,按提示下载即可:)


2. 编译脚本配置

注意将脚本中的 export NDK 换成自己的路径

 #!/bin/bash

    echo ">>>>>>>>> 编译硬件解码版本 <<<<<<<<"
    #替换为你自己的NDK路径.
    export NDK=/home/xaye/Android/Sdk/ndk/android-ndk-r22
    TOOLCHAIN=$NDK/toolchains/llvm/prebuilt/linux-x86_64


    function build_android
    {


    echo "开始编译 $CPU"

    ./configure \
    --prefix=$PREFIX \
    --enable-neon  \
    --enable-hwaccels  \
    --enable-gpl   \
    --enable-postproc \
    --enable-shared \
    --disable-debug \
    --enable-small \
    --enable-jni \
    --enable-mediacodec \
    --enable-decoder=h264_mediacodec \
    --disable-static \
    --disable-doc \
    --enable-ffmpeg \
    --disable-ffplay \
    --disable-ffprobe \
    --disable-avdevice \
    --disable-doc \
    --disable-symver \
    --cross-prefix=$CROSS_PREFIX \
    --target-os=android \
    --arch=$ARCH \
    --cpu=$CPU \
    --cc=$CC \
    --cxx=$CXX \
    --enable-cross-compile \
    --sysroot=$SYSROOT \
    --extra-cflags="-Os -fpic $OPTIMIZE_CFLAGS" \
    --extra-ldflags="$ADDI_LDFLAGS"


    make clean
    make
    make install

    echo "编译成功 $CPU"

    }

    #armv8-a
    ARCH=arm64
    CPU=armv8-a
    API=21
    CC=$TOOLCHAIN/bin/aarch64-linux-android$API-clang
    CXX=$TOOLCHAIN/bin/aarch64-linux-android$API-clang++
    SYSROOT=$NDK/toolchains/llvm/prebuilt/linux-x86_64/sysroot
    CROSS_PREFIX=$TOOLCHAIN/bin/aarch64-linux-android-
    PREFIX=$(pwd)/android/$CPU
    OPTIMIZE_CFLAGS="-march=$CPU"

    build_android

    #armv7-a
    ARCH=arm
    CPU=armv7-a
    API=16
    CC=$TOOLCHAIN/bin/armv7a-linux-androideabi$API-clang
    CXX=$TOOLCHAIN/bin/armv7a-linux-androideabi$API-clang++
    SYSROOT=$NDK/toolchains/llvm/prebuilt/linux-x86_64/sysroot
    CROSS_PREFIX=$TOOLCHAIN/bin/arm-linux-androideabi-
    PREFIX=$(pwd)/android/$CPU
    OPTIMIZE_CFLAGS="-mfloat-abi=softfp -mfpu=vfp -marm -march=$CPU "

    build_android

打开 FFmpeg 根目录的 configure 文件,搜索关键字 cc_default="clang"clang 改为 gcc,不修改的话编译会报错!

//1. 修改 configure 文件
vim configure
//2. 把 默认的 clang 修改为 gcc
if test "$target_os" = android; then
   # cc_default="clang"
     cc_default="gcc"
fi

作用:

  1. 绕过 configure 对 Clang 的严格检测。
  2. 利用 NDK 的 gcc 脚本的兼容性层,解决某些编译问题。

这一修改是针对旧版 FFmpeg 在 NDK r22 环境下的临时解决方案,本质是利用 NDK 的 gcc 包装器间接调用 Clang


3. 执行编译

# 赋予执行权限
chmod +x build_android.sh

# 开始编译(约10-30分钟)
./build_android.sh

编译完成后,会生成 android/armv7-aandroid/armv8-a 目录,结构如下

android/
    ├── armv7-a/
    │   ├── lib/*.so
    │   └── include/   <-- FFmpeg 头文件
    └── armv8-a/
        ├── lib/*.so
        └── include/

这些 .so 文件,分别对应:

  • armv7-a/ → 用于 Android 项目的 armeabi-v7a
  • armv8-a/ → 用于 Android 项目的 arm64-v8a

4. Android 项目配置

修改 build.gradle,启用 NDK 支持

 android {
        defaultConfig {
            externalNativeBuild {
                cmake {
                    cppFlags "-std=c++11"
                    abiFilters 'armeabi-v7a', 'arm64-v8a'
                }
            }
        }

        externalNativeBuild {
            cmake {
                path "CMakeLists.txt"
            }
        }

        ndkVersion '22.0.7026061'
    }

放入头文件

将 FFmpeg 头文件复制到 app/src/main/cpp/ 下,就是把上面编译生成的整个 include 文件夹复制进去,不用在意v7a还是v8a,头文件接口都是一样的。

创建 CMakeLists.txt

app/ 目录下创建:

# 最低 CMake 版本要求
cmake_minimum_required(VERSION 3.4.1)

# 项目设置
project("ffmpeg_jni")

# 设置 C 标准(FFmpeg 需要 C11)
set(CMAKE_C_STANDARD 11)
set(CMAKE_CXX_STANDARD 11)

# 打印当前 ABI 用于调试
message("Current ABI: ${ANDROID_ABI}")

# 设置 FFmpeg 库路径(根据实际路径调整)
set(FFMPEG_LIBS_DIR ${CMAKE_SOURCE_DIR}/src/main/jniLibs/${ANDROID_ABI})

# 定义 FFmpeg 核心库(按依赖顺序)
set(FFMPEG_LIBS
        avutil
        swresample
        avcodec
        avformat
        swscale
        # 可选添加其他库:postproc, avfilter 等
)

# 导入预编译的 FFmpeg 共享库
foreach(LIB ${FFMPEG_LIBS})
    add_library(${LIB} SHARED IMPORTED)
    set_target_properties(${LIB} PROPERTIES
            IMPORTED_LOCATION "${FFMPEG_LIBS_DIR}/lib${LIB}.so"
            # 对于 Android 8.0+ 需要设置 SONAME
            IMPORTED_SONAME "lib${LIB}.so"
    )
    message("Imported lib: ${FFMPEG_LIBS_DIR}/lib${LIB}.so")
endforeach()

# 添加 Android 日志库
find_library(log-lib log)

# 设置 JNI 源文件
file(GLOB JNI_SOURCES src/main/cpp/*.cpp)

# 创建 JNI 库
add_library(ffmpeg_jni SHARED ${JNI_SOURCES})

# 头文件包含路径(根据 FFmpeg 头文件位置调整)
target_include_directories(ffmpeg_jni PRIVATE
        ${CMAKE_SOURCE_DIR}/src/main/cpp/include  # 头文件放在这里
        ${CMAKE_SOURCE_DIR}/src/main/jniLibs/${ANDROID_ABI}/include  # 如果 FFmpeg 头文件随库提供
)

# 链接库
target_link_libraries(ffmpeg_jni
        android
        ${log-lib}
        ${FFMPEG_LIBS}  # 按依赖顺序自动链接
)

# 编译选项优化
target_compile_options(ffmpeg_jni PRIVATE
        -Wall
        -Werror
        -fno-exceptions
        -fno-rtti
        -fvisibility=hidden
)

5. JNI 代码实现

Java 层声明

创建 FFmpegHelper.java

public class FFmpegHelper {
    static {
        // 按依赖顺序加载FFmpeg库
        System.loadLibrary("avutil");
        System.loadLibrary("swresample");
        System.loadLibrary("avcodec");
        System.loadLibrary("avformat");
        System.loadLibrary("swscale");
        System.loadLibrary("ffmpeg_jni"); // 我们的JNI库
    }

    public static native String getFFmpegVersion();
}

Native 层实现

创建 ffmpeg_jni.cpp

#include <jni.h>
#include <android/log.h>
//#include <libavutil/avutil.h>
#include <stdio.h>

extern "C" {
#include <libavutil/avutil.h>
}

#define LOG_TAG "FFmpegJNI"
#define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG, LOG_TAG, __VA_ARGS__)



extern "C" {
JNIEXPORT jstring JNICALL
Java_com_xaye_compiler_FFmpegHelper_getFFmpegVersion(JNIEnv *env, jclass clazz) {
    // 调用FFmpeg API获取版本信息
    const char* version = av_version_info();

    LOGD("Native FFmpeg version: %s", version);

    return env->NewStringUTF(version ? version : "Unknown");
}
}

注意:在#include ffmpeg 库的头文件时,要使用 extern "C" 包起来,不然会报错!


6. 验证

在主界面 打印版本号

  Log.i("MainActivity", " FFmpeg version : "+FFmpegHelper.getFFmpegVersion());

输出:

在这里插入图片描述


代码已上传 ffmpeg-compiler

参考:音视频学习 (六) 一键编译 32/64 位 FFmpeg 4.2.2

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2375475.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

如何使用 Qwen3 实现 Agentic RAG?

今天&#xff0c;我们将学习如何部署由阿里巴巴最新Qwen 3驱动的Agentic RAG。 这里是我们的工具栈&#xff1a; CrewAI用于代理编排。 Firecrawl用于网络搜索。 LightningAI的LitServe用于部署。 顶部的视频展示了这一过程。 图表显示了我们的Agentic RAG流程&#xff1…

相机、雷达标定工具,以及雷达自动标定的思路

本篇我们来看一下自动驾驶传感器配置一个非常重要的模块&#xff0c;也就是传感器的标定。这里主要是对我之前修改的功能包的使用进行一个介绍. 对应的资源也已经上传了&#xff0c;0积分下载 安装 首先整个项目是使用ros1来进行启动的,但是要想正常编译,需要先安装三个对应的…

vsomeip环境搭建保姆级教程

vsomeip环境搭建保姆级教程 ubuntu环境搭建 {% links %} site: VMware搭建ubuntu保姆级教程 url: https://zhuanlan.zhihu.com/p/1903219373906327339 desc: flechazo image: https://q1.qlogo.cn/g?b=qq&nk=2861099&s=5 color: “#9d5b8b” {% endlinks %} vsomei…

我的MCP相关配置记录

1.VSCode的Cline中的MCP {"mcpServers": {"github.com/modelcontextprotocol/servers/tree/main/src/github": {"autoApprove": [],"disabled": false,"timeout": 60,"command": "cmd","args&quo…

我们来学nacos -- 集群nacos2.5.1mysql8.4

2.5.1集群搭建 架构下载解压到3个文件夹初始化数据库&数据迁移检查端口可用配置cluster.confapplication.properties 使用mysql8.4的jar启动db.num is null报错datasource错误成功 nginx反向代理集群查看 架构 其中包含3个nacos节点&#xff0c;然后一个负载均衡器代理3个…

Rollup入门与进阶:为现代Web应用构建超小的打包文件

我们常常面临Webpack复杂配置或是Babel转译后的冗余代码&#xff0c;结果导致最终的包体积居高不下加载速度也变得异常缓慢&#xff0c;而在众多打包工具中Rollup作为一个轻量且高效的选择&#xff0c;正悄然改变着这一切&#xff0c;本文将带你深入了解这个令人惊艳的打包工具…

专题四:综合练习( 找出所有子集的异或总和再求和)

以leetcode1863题为例 题目分析&#xff1a; 找到每个子集&#xff0c;然后子集中的元素异或之后全部相加 算法原理分析&#xff1a; 画决策树&#xff1a;第一层为这个子集有一个元素 第二层这个子集有两个元素 从上往下罗列&#xff0c;把所有子集都罗列出来&#xf…

STM32 修炼手册

第一章 计算机体系结构(了解) 后续在板子上开发的时候&#xff0c;需要考虑是否有操作系统 方式一&#xff1a;有操作系统&#xff0c;通过c库通过os api操作硬件方式二&#xff1a;无操作系统&#xff0c; 通过c库通过固件库操作硬件 第二章 STM32开发板概述 板子/开发板&…

缓存(2):数据一致性

概述 一致性就是数据保持一致,在分布式系统中,可以理解为多个节点中数据的值是一致的。 强一致性:这种一致性级别是最符合用户直觉的,它要求系统写入什么,读出来的也会是什么,用户体验好,但实现起来往往对系统的性能影响大弱一致性:这种一致性级别约束了系统在写入成功…

ppy/osu构建

下载 .NET (Linux、macOS 和 Windows) | .NET dotnet还行 构建&#xff1a;f5 运行&#xff1a;dotnet run --project osu.Desktop -c Debug

基于几何布朗运动的股价预测模型构建与分析

基于几何布朗运动的股价预测模型构建与分析 摘要 本文建立基于几何布朗运动的股价预测模型&#xff0c;结合极大似然估计与蒙特卡洛模拟&#xff0c;推导股价条件概率密度函数并构建动态预测区间。实证分析显示模型在标普500指数预测中取得89%的覆盖概率&#xff0c;波动率估…

python如何提取Chrome中的保存的网站登录用户名密码?

很多浏览器都贴心地提供了保存用户密码功能&#xff0c;用户一旦开启&#xff0c;就不需要每次都输入用户名、密码&#xff0c;非常方便。作为python脚本&#xff0c;能否拿到用户提前保存在浏览器中的用户名密码&#xff0c;用以自动登录呢&#xff1f;必须有&#xff0c;小爬…

Redis实现分布式获取全局唯一自增ID的案例。

【1】简易自增版本(从 1 开始 1,2,3&#xff0c;...) 项目结构 下面是一个基于 RedisTemplate 实现的分布式全局唯一自增 ID 生成器的案例。适用于 Java Spring Boot 环境&#xff0c;利用 Redis 的原子操作 INCR 指令。 ✅ 原理说明 Redis 提供的 INCR 命令是原子性的&…

人脸识别备案:筑牢人脸信息 “安全墙”

人脸识别备案制度主要依据《人脸识别技术应用安全管理办法》建立&#xff0c;人脸识别技术广泛应用于安防、金融、门禁、交通等领域&#xff0c;带来便利高效的同时&#xff0c;人脸信息安全问题也引发担忧。为规范技术应用、保护个人信息权益&#xff0c;人脸识别备案制度应运…

基于RT-Thread的STM32F4开发第三讲——DAC

文章目录 前言一、DAC是什么&#xff1f;二、RT-Thread工程创建三、DAC函数编写1.DAC.c2.DAC.h3.main.c 四、结果测试五、工程分享 前言 本章利用RT-Thread最新的驱动5.1.0开发DAC模块&#xff0c;使用的开发板是正点原子的STM32F4探索者。很多配置和上文重复&#xff0c;本文…

网络状态可以通过hutool.HttpStatus获取

网络状态可以通过hutool.HttpStatus获取 全部都是静态int类型

Gemini 2.5 推动视频理解进入新时代

每周跟踪AI热点新闻动向和震撼发展 想要探索生成式人工智能的前沿进展吗&#xff1f;订阅我们的简报&#xff0c;深入解析最新的技术突破、实际应用案例和未来的趋势。与全球数同行一同&#xff0c;从行业内部的深度分析和实用指南中受益。不要错过这个机会&#xff0c;成为AI领…

谈谈各种IO模型

目前的IO模型有5种&#xff1a;BIO&#xff08;阻塞IO&#xff09;、NIO&#xff08;非阻塞IO&#xff09;、IO多路复用、信号驱动IO、异步IO&#xff08;AIO&#xff09; 了解这些模型之前&#xff0c;我们需要先知道IO模型中的几个概念&#xff1a;阻塞&非阻塞、同步&am…

Linux系统管理与编程20:Apache

兰生幽谷&#xff0c;不为莫服而不芳&#xff1b; 君子行义&#xff0c;不为莫知而止休。 做好网络和yum配置&#xff0c;用前面dns规划的www的IP进行。 #!/bin/bash #----------------------------------------------------------- # File Name: myWeb.sh # Version: 1.0 # …

BFS算法篇——打开智慧之门,BFS算法在拓扑排序中的诗意探索(下)

文章目录 引言一、课程表1.1 题目链接&#xff1a;https://leetcode.cn/problems/course-schedule/description/1.2 题目分析&#xff1a;1.3 思路讲解&#xff1a;1.4 代码实现&#xff1a; 二、课程表||2.1 题目链接&#xff1a;https://leetcode.cn/problems/course-schedul…