FFmpeg中使用Android Content协议打开文件设备

引言

随着Android 10引入的Scoped Storage(分区存储)机制,传统的文件访问方式发生了重大变化。FFmpeg作为强大的多媒体处理工具,也在不断适应Android平台的演进。本文将介绍如何在FFmpeg 7.0+版本中使用Android content协议直接访问文件,为开发者提供更便捷的多媒体处理方案。

需要说明的是,本文记录的是我个人的实践经验,并非官方文档。由于相关技术较新,网络上的参考资料有限,如有错误疏漏,还请大家指点。

背景:Scoped Storage与文件访问挑战

自Android 10(API 29)起,Google实施了Scoped Storage策略,即使应用拥有READ_EXTERNAL_STORAGE权限,也无法直接通过/sdcard/路径访问文件。开发者必须使用Storage Access Framework (SAF)获取用户选择文件的content URI。这给多媒体处理带来了新的挑战。

传统方案:文件描述符(fd)协议

在FFmpeg支持content协议之前,开发者通常采用以下工作流程:

  1. 通过SAF获取文件URI
  2. 转换为文件描述符(fd)
  3. 通过FFmpeg的fd协议处理文件
// Java端实现
private int getFileDescriptor(Uri uri) {try (ParcelFileDescriptor pfd = getContentResolver().openFileDescriptor(uri, "r")) {return pfd != null ? pfd.detachFd() : -1;} catch (IOException e) {Log.e(TAG, "Error opening file descriptor", e);return -1;}
}

这种方案存在以下局限性:

  • 需要额外的Java层转换代码
  • 文件描述符管理复杂
  • 不支持直接URI访问
  • 跨进程传递文件描述符存在兼容性问题

现代化方案:Content协议支持

2024年2月,FFmpeg正式合并了对Android content协议的支持(提交记录:6567516a5ef),开发者现在可以直接使用content URI进行多媒体处理。

环境配置

1. 初始化JNI环境

在JNI加载时设置Java虚拟机:

#include <jni.h>extern "C" JNIEXPORT jint JNI_OnLoad(JavaVM* vm, void* reserved) {av_jni_set_java_vm(vm, nullptr);return JNI_VERSION_1_6;  // 建议使用较新的JNI版本
}
2. 传递应用上下文
// MainActivity.java
public native void initFFmpeg(Context context);// 在Activity初始化时调用
initFFmpeg(getApplicationContext());

对应的JNI实现:

extern "C" JNIEXPORT void JNICALL
Java_com_example_media_MainActivity_initFFmpeg(JNIEnv* env,jobject thiz,jobject context) {// 创建全局引用防止被GC回收jobject global_ctx = env->NewGlobalRef(context);av_jni_set_android_app_ctx(global_ctx, nullptr);
}

使用Content协议

配置完成后,可以直接将SAF获取的URI传递给FFmpeg:

@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {if (requestCode == REQUEST_PICK_VIDEO && resultCode == RESULT_OK&& data != null) {Uri uri = data.getData();String ffmpegUrl = uri.toString();// 传递给FFmpeg处理processMediaWithFFmpeg(ffmpegUrl);}
}

结论

FFmpeg对Android content协议的支持显著简化了在Scoped Storage环境下的多媒体处理流程。开发者现在可以:

  • 直接使用SAF获取的URI
  • 减少Java层转换代码
  • 获得更好的内存管理
  • 保持与Android最新存储策略的兼容性

建议新项目优先采用content协议方案,既符合Android最佳实践,又能简化开发流程。对于需要支持旧版FFmpeg的项目,可暂时保留fd协议作为fallback方案。

参考资料

  1. FFmpeg官方提交记录
  2. Android Scoped Storage文档
  3. Storage Access Framework指南

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

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

相关文章

vue——v-pre的使用

&#x1f530; 基础理解 ✅ 什么是 v-pre&#xff1f; v-pre 是一个跳过编译的 Vue 指令。 它告诉 Vue&#xff1a;“这个元素和其子元素中的内容不要被编译处理&#xff0c;按原样输出。” ✅ 使用场景&#xff1a; 展示原始的 Mustache 插值语法&#xff08;{{ xxx }}&a…

PyTorch中TensorBoardX模块与torch.utils.tensorboard模块的对比分析

文章目录 说明1. 模块起源与开发背景2. 功能特性对比3. 安装与依赖关系4. 性能与使用体验5. 迁移与兼容性策略6. 最佳实践与建议7. 未来展望8. 结论实际相关信息推荐资源 说明 TensorBoard&#xff1a;独立工具&#xff0c;只需安装tensorboard。TensorFlow&#xff1a;非必需…

单片机中断系统工作原理及定时器中断应用

文件目录 main.c #include <REGX52.H> #include "TIMER0.H" #include "KEY.H" #include "DELAY.H"//void Timer0_Init() { // TMOD 0x01; // TL0 64536 % 256; // TH0 64536 / 256; // ET0 1; // EA 1; // TR0 1; //}unsigned char…

Python爬虫实战:研究Portia框架相关技术

1. 引言 1.1 研究背景与意义 在大数据时代,网络数据已成为企业决策、学术研究和社会分析的重要资源。据 Statista 统计,2025 年全球数据总量将达到 175ZB,其中 80% 以上来自非结构化网络内容。如何高效获取并结构化这些数据,成为数据科学领域的关键挑战。 传统爬虫开发需…

【机器学习基础】机器学习与深度学习概述 算法入门指南

机器学习与深度学习概述 算法入门指南 一、引言&#xff1a;机器学习与深度学习&#xff08;一&#xff09;定义与区别&#xff08;二&#xff09;发展历程&#xff08;三&#xff09;应用场景 二、机器学习基础&#xff08;一&#xff09;监督学习&#xff08;二&#xff09;无…

[C语言初阶]扫雷小游戏

目录 一、原理及问题分析二、代码实现2.1 分文件结构设计2.2 棋盘初始化与打印2.3 布置雷与排查雷2.4 游戏主流程实现 三、后期优化方向 在上一篇文章中&#xff0c;我们实现了我们的第二个游戏——三子棋小游戏。这次我们继续结合我们之前所学的所有内容&#xff0c;制作出我们…

ROS云课三分钟-破壁篇GCompris-一小部分支持Edu应用列表-2025

开启蓝桥云课ROS ROS 机器人操作系统初级教程_ROS - 蓝桥云课 安装和使用GCompris 终端输入&#xff1a;sudo apt install gcompris sudo apt install gcompris ok&#xff0c;完成即可。 sudo apt install gcompris 如果是平板&#xff0c;秒变儿童学习机。 启动 流畅运…

Linux系统基础——是什么、适用在哪里、如何选

一、Linux是什么 Linux最初是由林纳斯托瓦兹&#xff08;Linus Torvalds&#xff09;基于个人兴趣爱好开发的个人项目&#xff0c;他编写了最核心的内核&#xff1b;后面为了发展壮大Linux系统他将整个项目开源到GitHub上&#xff0c;可以让全世界的人都参与到项目的开发维护中…

26、AI 预测性维护 (燃气轮机轴承) - /安全与维护组件/ai-predictive-maintenance-turbine

76个工业组件库示例汇总 AI 预测性维护模拟组件 (燃气轮机轴承) 概述 这是一个交互式的 Web 组件,旨在模拟基于 AI 的预测性维护 (Predictive Maintenance, PdM) 概念,应用于工业燃气轮机的关键部件(例如轴承)。它通过模拟传感器数据、动态预测剩余使用寿命 (RUL),并根…

el-form 使用el-row el-col对齐 注意事项

1.el-form 使用inline&#xff0c;el-form-item宽度会失效。 2.为了保证el-form-item 和 它内部的el-input 能在一行&#xff0c;要设置el-form-item的label-width <el-form :model"editInspectform"><el-row style"margin-bottom: 20px"><…

mac 安装 mysql 和 mysqlshell

1. 安装 mysql https://dev.mysql.com/downloads/mysql/?spma2c6h.12873639.article-detail.4.37474f4dTHdszC 默认mysql未配置环境变量&#xff0c;可以在设置中找到 2. 安装 mysqlshell https://dev.mysql.com/downloads/shell/ #启动mysql-shell mysqlsh 3. 使用 mysq…

漏洞检测与渗透检验在功能及范围上究竟有何显著差异?

漏洞检测与渗透检验是确保系统安全的重要途径&#xff0c;这两种方法各具特色和功效&#xff0c;它们在功能上有着显著的差异。 目的不同 漏洞扫描的主要任务是揭示系统内已知的安全漏洞和隐患&#xff0c;这就像是对系统进行一次全面的健康检查&#xff0c;看是否有已知的疾…

机器学习模型度量指标(混淆矩阵、准确率、精确率、召回率、F1分数、ROC曲线、AUC、平均精度均值)

我们研究的是多分类问题&#xff0c;下面所有例子以多分类问题举例 混淆矩阵&#xff08;Confusion Matrix&#xff09; 混淆矩阵&#xff08; Confusion Matrix &#xff09;是一个表格&#xff0c;用于可视化机器学习模型在分类问题上 的性能。混淆矩阵的行表示实际类别&…

打卡day35

一、模型结构可视化 理解一个深度学习网络最重要的2点&#xff1a; 了解损失如何定义的&#xff0c;知道损失从何而来----把抽象的任务通过损失函数量化出来了解参数总量&#xff0c;即知道每一层的设计才能退出—层设计决定参数总量 为了了解参数总量&#xff0c;我们需要知…

时序数据库 TDengine × Superset:一键构建你的可视化分析系统

如果你正在用 TDengine 管理时序数据&#xff0c;写 SQL 查询没问题&#xff0c;但一到展示环节就犯难——图表太基础&#xff0c;交互不够&#xff0c;甚至连团队都看不懂你辛苦分析的数据成果&#xff1f;别担心&#xff0c;今天要介绍的这个组合&#xff0c;正是为你量身打造…

C# 初学者的 3 种重构模式

(Martin Fowlers Example) 1. 积极使用 Guard Clause&#xff08;保护语句&#xff09; "如果条件不满足&#xff0c;立即返回。将核心逻辑放在最少缩进的地方。" 概念定义 Guard Clause&#xff08;保护语句&#xff09; 是一种在函数开头检查特定条件是否满足&a…

基于51单片机和8X8点阵屏、独立按键的滑动躲闪类小游戏

目录 系列文章目录前言一、效果展示二、原理分析三、各模块代码1、8X8点阵屏2、独立按键3、定时器04、定时器1 四、主函数总结 系列文章目录 前言 用的是普中A2开发板。 【单片机】STC89C52RC 【频率】12T11.0592MHz 【外设】8X8点阵屏、独立按键 效果查看/操作演示&#xff…

Java面向对象 一

系列文章目录 Java面向对象 二-CSDN博客 Java面向对象 三-CSDN博客 目录 系列文章目录 前言 一、初步认识面向对象 1.类和对象的简单理解 2.类的构成 二、类的实例化 1.对象的创建 2.对象的初始化 三、this引用的作用 四、构造方法 1.构造方法的提供 2.对象的构…

深度学习Y8周:yolov8.yaml文件解读

&#x1f368; 本文为&#x1f517;365天深度学习训练营中的学习记录博客&#x1f356; 原作者&#xff1a;K同学啊 本周任务&#xff1a;根据yolov8n、yolov8s模型的结构输出&#xff0c;手写出yolov8l的模型输出、 文件位置&#xff1a;./ultralytics/cfg/models/v8/yolov8.…

【RocketMQ 生产者和消费者】- 生产者启动源码 - MQClientInstance 定时任务(4)

文章目录 1. 前言2. startScheduledTask 启动定时任务2.1 fetchNameServerAddr 拉取名称服务地址2.2 updateTopicRouteInfoFromNameServer 更新 topic 路由信息2.2.1 topic 路由信息2.2.2 updateTopicRouteInfoFromNameServer 获取 topic2.2.3 updateTopicRouteInfoFromNameSer…