当前位置: 首页 > news >正文

从零到一学FFmpeg:avformat_alloc_output_context2 函数详析与实战

文章目录

  • 前言
  • 一、函数原型
  • 二、功能描述
  • 三、使用场景
  • 四、AVFormatContext 结构体
  • 五、代码实例


前言

avformat_alloc_output_context2 是FFmpeg库中的一个函数,用于为输出多媒体文件初始化一个AVFormatContext结构体。这个函数在开始输出音频、视频数据到文件之前被调用,它是多媒体文件操作过程中的一个关键步骤。


提示:以下是本篇文章正文内容,下面案例可供参考

一、函数原型

int avformat_alloc_output_context2(AVFormatContext **ctx, const AVOutputFormat *oformat,const char *format_name,const char *filename);

参数说明

1、**AVFormatContext **ctx: 这是一个输出上下文指针的指针。
函数会为输出操作分配并初始化一个AVFormatContext结构,并将指针地址赋给ctx。
在调用此函数前,应将ctx初始化为NULL。2、const AVOutputFormat *oformat: 指向输出格式的指针。
这个参数可以是NULL,此时函数会根据format_name或filename的扩展名自动推断输出格式。
如果不为NULL,你应该提供一个特定的输出格式
(例如,对于MP4文件,可能是av_guess_format("mp4", NULL, NULL)的结果),这允许你精确控制输出格式。3、const char *format_name: 一个指定输出格式的字符串。
如果oformat为NULL且format_name非空,FFmpeg会尝试根据这个名字找到合适的输出格式。
例如,你可以传入"mp4"来指定输出为MP4格式。4、const char *filename: 输出文件的名称。
这个参数主要用于根据文件扩展名自动生成输出格式(当oformat为NULL时)。
如果同时提供了format_name,则此参数主要用于信息提示,并帮助确定编码器等。

返回值

成功时返回0。
失败时返回一个负的错误代码(如AVERROR_NOENT、AVERROR_INVALIDDATA等),
具体可以通过av_strerror()函数转化为可读的错误信息

二、功能描述

分配与初始化: 分配内存给AVFormatContext结构体并进行基本的初始化设置,准备用于输出多媒体流。
自动检测格式: 如果未直接指定输出格式(oformat为NULL),函数会根据提供的文件名或格式名称自动检测应使用的输出格式。
准备输出: 为后续的音频、视频流的封装和实际数据的写入做准备

三、使用场景

当你需要使用FFmpeg库编写程序以编码并输出音频、视频到文件时,这个函数是必须的初始化步骤。之后,你通常会继续使用其他FFmpeg函数添加音视频流、打开文件、写入头信息、发送编码好的数据包等。

请注意,使用完毕后,应该通过avio_closep(&ctx->pb)关闭IO上下文,并通过avformat_free_context(ctx)释放整个AVFormatContext结构体所占用的资源。

四、AVFormatContext 结构体

AVFormatContext是FFmpeg库中的一个核心结构体,它用于存储与多媒体容器格式相关的全局信息以及对输入输出操作的上下文管理。当你处理音频、视频文件或者流媒体时,无论是进行解复用(demuxing)还是复用(muxing),都会用到这个结构体。

结构体定义概述:

typedef struct AVFormatContext {/* 基本信息 */AVClass *av_class;           // 对象的类信息,用于日志和运行时类型信息intiformat *iformat;         // 输入格式上下文,包含文件格式信息AVOutputFormat *oformat;     // 输出格式上下文,包含输出文件的格式信息/* 文件/网络IO相关 */AVIOContext *pb;             // 输入输出缓冲区上下文,处理底层的读写操作char *filename;              // 当前操作的文件名或URL/* 时间基相关 */AVRational pkt_timebase;     // 数据包时间基,用于时间戳转换int64_t start_time;          // 流开始的时间戳int64_t duration;            // 流的总时长,单位是AV_TIME_BASE时间单位/* 音视频流信息 */int nb_streams;              // 流的数量AVStream **streams;           // 指向AVStream结构体数组的指针,每个AVStream对应一个音视频流/* 其他控制选项 */int flags;                   // 格式上下文的标志,比如AVFMT_FLAG_GENPTSint probesize;               // 分析文件头时读取的数据量上限int max_analyze_duration;    // 最大分析时长,防止分析无限进行.../* 更多字段,这里省略了众多用于控制和状态跟踪的高级选项 */
} AVFormatContext;

五、代码实例

avformat_alloc_output_context2 主要用于为输出多媒体文件或流创建并初始化一个AVFormatContext结构体实例,下面是一个使用该函数的代码示例,展示了如何准备输出上下文以便进行复用(即封装编码后的音视频数据到一个文件中):

#include <libavformat/avformat.h>
#include <libavcodec/avcodec.h>
#include <stdio.h>int main() {// 初始化FFmpeg库av_register_all();avformat_network_init();const char *output_filename = "output.mp4"; // 输出文件名const char *format_name = NULL; // 可以指定输出格式名称,如果根据文件名能自动推断则无需指定AVFormatContext *output_format_ctx = NULL; // 输出格式上下文指针// 使用avformat_alloc_output_context2创建输出上下文int ret = avformat_alloc_output_context2(&output_format_ctx, NULL, format_name, output_filename);if (ret < 0) {printf("Could not create output context\n");return -1;}// 如果是根据文件名自动推断的输出格式,则可以从output_format_ctx->oformat中获取实际的输出格式信息if (!format_name) {printf("Automatically detected output format: %s\n", output_format_ctx->oformat->name);} else {// 如果手动指定了format_name,则此处可以进行额外的格式设置或验证}// 接下来通常需要添加音视频流到输出上下文中,配置流的相关参数,// 然后打开输出文件,并开始写入头部信息、编码数据等操作。// ...此处省略添加流、打开输出文件、写入数据等后续步骤...// 最终,记得清理资源if (output_format_ctx != NULL) {avio_close(output_format_ctx->pb);avformat_free_context(output_format_ctx);}return 0;
}

相关文章:

  • 【for循环】最大跨度
  • C语言入门系列:指针入门(超详细)
  • Maven添加reactor依赖失败
  • DS:堆的应用——两种算法和TOP-K问题
  • 为什么print语句被Python3遗弃?
  • 玄机——第六章 流量特征分析-waf 上的截获的黑客攻击流量 wp
  • Java学习笔记(一)Java内容介绍、程序举例、DOS命令、Java跨平台特性的本质、课后练习
  • Lua 面向对象编程
  • 如何修改外接移动硬盘的区号
  • 【RabbitMQ】异步消息及Rabbitmq安装
  • ardupilot开发 --- Jetson Orin Nano 后篇
  • 阿里云 邮件系统DNS域名解析 搭配 postfix+dovecot 邮件服务器
  • 打印水仙花数
  • 数据挖掘与分析——数据预处理
  • IMU用于飞行坐姿校正
  • HashMap剖析之内部结构
  • JavaScript 无符号位移运算符 三个大于号 的使用方法
  • magento 货币换算
  • PermissionScope Swift4 兼容问题
  • Travix是如何部署应用程序到Kubernetes上的
  • 简单数学运算程序(不定期更新)
  • 通过获取异步加载JS文件进度实现一个canvas环形loading图
  • 小程序上传图片到七牛云(支持多张上传,预览,删除)
  • 原生 js 实现移动端 Touch 滑动反弹
  • 原生Ajax
  • Unity3D - 异步加载游戏场景与异步加载游戏资源进度条 ...
  • 东超科技获得千万级Pre-A轮融资,投资方为中科创星 ...
  • ​猴子吃桃问题:每天都吃了前一天剩下的一半多一个。
  • ​如何使用QGIS制作三维建筑
  • # Swust 12th acm 邀请赛# [ K ] 三角形判定 [题解]
  • # 职场生活之道:善于团结
  • #java学习笔记(面向对象)----(未完结)
  • (1)虚拟机的安装与使用,linux系统安装
  • (4.10~4.16)
  • (C语言)球球大作战
  • (Mac上)使用Python进行matplotlib 画图时,中文显示不出来
  • (附源码)ssm跨平台教学系统 毕业设计 280843
  • (黑马出品_高级篇_01)SpringCloud+RabbitMQ+Docker+Redis+搜索+分布式
  • (论文阅读32/100)Flowing convnets for human pose estimation in videos
  • (每日持续更新)jdk api之StringBufferInputStream基础、应用、实战
  • (入门自用)--C++--抽象类--多态原理--虚表--1020
  • (太强大了) - Linux 性能监控、测试、优化工具
  • (五)网络优化与超参数选择--九五小庞
  • (一)插入排序
  • (转)h264中avc和flv数据的解析
  • (转)Linux NTP配置详解 (Network Time Protocol)
  • (转)真正的中国天气api接口xml,json(求加精) ...
  • *算法训练(leetcode)第四十五天 | 101. 孤岛的总面积、102. 沉没孤岛、103. 水流问题、104. 建造最大岛屿
  • .net core 的缓存方案
  • .NET Core、DNX、DNU、DNVM、MVC6学习资料
  • .NET Remoting Basic(10)-创建不同宿主的客户端与服务器端
  • .net 程序 换成 java,NET程序员如何转行为J2EE之java基础上(9)
  • .net 使用$.ajax实现从前台调用后台方法(包含静态方法和非静态方法调用)
  • .net快速开发框架源码分享
  • .NET连接MongoDB数据库实例教程