FFmpeg AAC文件和H264文件合成MP4/FLV文件

使用FFmpeg库把AAC文件和H264文件合成MP4/FLV文件,FFmpeg版本为4.4.2-0。

需要aac和h264测试文件的,可以从我上传的MP4文件中用ffmpeg提取,命令如下:

ffmpeg -i <input.mp4> -map 0:v -c:v copy <output.h264> -map 0:a -c:a copy <output.aac>

代码如下:
 

#include <stdio.h>
#include "libavformat/avformat.h"// 打开输入文件并查找流信息
int open_input_file(const char *filename, AVFormatContext **ifmt_ctx)
{// 打开输入文件if (avformat_open_input(ifmt_ctx, filename, 0, 0) < 0){fprintf(stderr, "open %s file failed\n", filename);return -1;}// 查找流信息if (avformat_find_stream_info(*ifmt_ctx, 0) < 0){fprintf(stderr, "avformat_find_stream_info failed\n");return -1;}return 0;
}// 创建输出流
int create_output_stream(AVFormatContext *ofmt_ctx, AVFormatContext *ifmt_ctx, enum AVMediaType type,int *index_in, int *index_out)
{for (unsigned int i = 0; i < ifmt_ctx->nb_streams; i++){// 查找指定类型的流if (ifmt_ctx->streams[i]->codecpar->codec_type == type){AVStream *out_stream = avformat_new_stream(ofmt_ctx, NULL);if (!out_stream){fprintf(stderr, "avformat_new_stream failed\n");return -1;}*index_in = i;*index_out = out_stream->index;// 复制输入流的编码参数到输出流if (avcodec_parameters_copy(out_stream->codecpar, ifmt_ctx->streams[i]->codecpar) < 0){fprintf(stderr, "avcodec_parameters_copy failed\n");return -1;}// 如果是音频流,设置相关标志if (type == AVMEDIA_TYPE_AUDIO){/** codec_tag 是一个标识符,用于指定特定的编解码器。* 将其设置为0表示在输出文件中不使用特定的编解码器标识符。*/out_stream->codecpar->codec_tag = 0;/** 检查输出格式的标志是否包含 AVFMT_GLOBALHEADER,* AVFMT_GLOBALHEADER 是一个标志,表示编解码器的头部信息应存储在文件的全局头部,* 而不是每个帧的头部。常用于某些格式(例如MP4),以减少每个帧的开销。*/if (ofmt_ctx->oformat->flags & AVFMT_GLOBALHEADER){/** 设置全局头部标志,如果输出格式需要全局头部,* 则在输出格式上下文的标志中添加 AV_CODEC_FLAG_GLOBAL_HEADER,* 这会通知编码器将头部信息写入文件的全局头部,而不是每个帧的头部。*/ofmt_ctx->flags |= AV_CODEC_FLAG_GLOBAL_HEADER;}}return 0;}}return -1;
}// 读取并写入帧
int rw_frame(AVFormatContext *ifmt_ctx, AVFormatContext *ofmt_ctx,int stream_index_in, int stream_index_out, int *frame_index, int64_t *cur_pts)
{int ret;AVPacket pkt;AVStream *in_stream, *out_stream;// 读取帧if ((ret = av_read_frame(ifmt_ctx, &pkt)) < 0){if (ret == AVERROR_EOF) // 读到文件尾{return -2;}else{fprintf(stderr, "av_read_frame failed\n");return -1;}}in_stream = ifmt_ctx->streams[pkt.stream_index];out_stream = ofmt_ctx->streams[stream_index_out];// 处理指定的流if (pkt.stream_index == stream_index_in){// 如果PTS值无效,计算并设置PTS和DTSif (pkt.pts == AV_NOPTS_VALUE){AVRational time_base = in_stream->time_base;int64_t calc_duration = (double)AV_TIME_BASE / av_q2d(in_stream->r_frame_rate);pkt.pts = (double)(*frame_index * calc_duration) / (double)(av_q2d(time_base) * AV_TIME_BASE);pkt.dts = pkt.pts;pkt.duration = (double)calc_duration / (double)(av_q2d(time_base) * AV_TIME_BASE);(*frame_index)++;}*cur_pts = pkt.pts;// 转换PTS和DTSpkt.pts = av_rescale_q_rnd(pkt.pts, in_stream->time_base, out_stream->time_base,(enum AVRounding)(AV_ROUND_NEAR_INF | AV_ROUND_PASS_MINMAX));pkt.dts = av_rescale_q_rnd(pkt.dts, in_stream->time_base, out_stream->time_base,(enum AVRounding)(AV_ROUND_NEAR_INF | AV_ROUND_PASS_MINMAX));pkt.duration = av_rescale_q(pkt.duration, in_stream->time_base, out_stream->time_base);pkt.pos = -1;pkt.stream_index = stream_index_out;if (in_stream->codecpar->codec_type == AVMEDIA_TYPE_VIDEO){printf("Write 1 video Packet. size:%d pts:%ld\n", pkt.size, pkt.pts);}else if (in_stream->codecpar->codec_type == AVMEDIA_TYPE_AUDIO){printf("Write 1 audio Packet. size:%d pts:%ld\n", pkt.size, pkt.pts);}// 写入帧if ((ret = av_interleaved_write_frame(ofmt_ctx, &pkt)) < 0){fprintf(stderr, "av_interleaved_write_frame failed\n");av_packet_unref(&pkt);return -1;}}av_packet_unref(&pkt);return 0;
}int main(int argc, char *argv[])
{int ret = -1, value = -1;const char *in_filename_v = argv[1];const char *in_filename_a = argv[2];const char *out_filename = argv[3];int videoindex_v = -1, videoindex_out = -1;int audioindex_a = -1, audioindex_out = -1;int frame_index = 0;int64_t cur_pts_v = 0, cur_pts_a = 0;int writing_v = 1, writing_a = 1;const AVOutputFormat *ofmt = NULL;AVFormatContext *ifmt_ctx_v = NULL, *ifmt_ctx_a = NULL, *ofmt_ctx = NULL;if (argc < 3){fprintf(stderr, "Usage: %s <h264 filename> <aac filename> <output filename>\n", argv[0]);return -1;}// 打开视频输入文件if (open_input_file(in_filename_v, &ifmt_ctx_v) < 0){goto end;}// 打开音频输入文件if (open_input_file(in_filename_a, &ifmt_ctx_a) < 0){goto end;}// 分配输出上下文avformat_alloc_output_context2(&ofmt_ctx, NULL, NULL, out_filename);if (!ofmt_ctx){fprintf(stderr, "avformat_alloc_output_context2 failed\n");goto end;}ofmt = ofmt_ctx->oformat;// 创建视频输出流if (create_output_stream(ofmt_ctx, ifmt_ctx_v, AVMEDIA_TYPE_VIDEO, &videoindex_v, &videoindex_out) < 0)goto end;// 创建音频输出流if (create_output_stream(ofmt_ctx, ifmt_ctx_a, AVMEDIA_TYPE_AUDIO, &audioindex_a, &audioindex_out) < 0)goto end;// 打开输出文件if (!(ofmt->flags & AVFMT_NOFILE)) // 检查输出格式是否需要文件存储{if (avio_open(&ofmt_ctx->pb, out_filename, AVIO_FLAG_WRITE)){fprintf(stderr, "open %s file failed\n", out_filename);goto end;}}// 写入文件头if (avformat_write_header(ofmt_ctx, NULL) < 0){fprintf(stderr, "avformat_write_header failed\n");goto end;}// 循环写入视频和音频帧while (writing_v || writing_a){// 如果还在写视频帧,且(不写音频帧或者视频帧的PTS小于等于音频帧的PTS)if (writing_v &&(!writing_a ||av_compare_ts(cur_pts_v, ifmt_ctx_v->streams[videoindex_v]->time_base, cur_pts_a,ifmt_ctx_a->streams[audioindex_a]->time_base) <= 0)){// 读取并写入视频帧value = rw_frame(ifmt_ctx_v, ofmt_ctx, videoindex_v,videoindex_out, &frame_index, &cur_pts_v);if (value == -2){writing_v = 0;}else if (value < 0){goto end;}}else{ // 读取并写入音频帧value = rw_frame(ifmt_ctx_a, ofmt_ctx, audioindex_a,audioindex_out, &frame_index, &cur_pts_a);if (value == -2){writing_a = 0;}else if (value < 0){goto end;}}}// 写入文件尾av_write_trailer(ofmt_ctx);ret = 0;end:if (ifmt_ctx_v)avformat_close_input(&ifmt_ctx_v);if (ifmt_ctx_a)avformat_close_input(&ifmt_ctx_a);if (ofmt && !(ofmt->flags & AVFMT_NOFILE))avio_close(ofmt_ctx->pb);if (ofmt_ctx)avformat_free_context(ofmt_ctx);return ret;
}

参考博客链接:FFMPEG库实现mp4/flv文件(H264+AAC)的封装与分离_ffmpeg mp4 flv-CSDN博客 

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

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

相关文章

【数据结构与算法(C语言)】循环队列图解

目录 1. 前言1.1 普通循环队列假溢出1.1.1 初始化队列1.1.2 插满队列1.1.3 删除元素后&#xff0c;再插入元素 1.2 循环队列1.2.1 插入元素&#xff0c;队列已满1.2.2 将元素J1、J2出列&#xff0c;循环队列又空出两个空间1.2.3 元素J6可以继续入列 2. 存储结构和函数说明2.1 队…

vivado PIP or SITE_PIP、PKGPIN_BYTEGROUP

PIP是Xilinx部件上用于路由连接或网络的设备对象。PIP 称为ARC的连接多路复用器可以编程为将一根电线连接到 另一个&#xff0c;从而将节点连接在一起&#xff0c;以形成中特定NET所需的路由 设计。 SITE_PIP&#xff0c;也称为路由BEL&#xff0c;是SITE内部的连接多路复用器&…

嵌入式跨平台编译:vsftpd

下载 https://security.appspot.com/vsftpd.html 或者直接下载&#xff1a; https://download.csdn.net/download/quantum7/89451093 解压 tar xf vsftpd-3.0.5.tar.gz 编译 export CROSS_NAMEaarch64-mix210-linux export PATH$PATH:/opt/linux/x86-arm/${CROSS_NAME}/…

力扣爆刷第153天之TOP100五连刷(相交、翻转、排序链表、螺旋矩阵、锯齿二叉树)

力扣爆刷第153天之TOP100五连刷&#xff08;相交、翻转、排序链表、螺旋矩阵、锯齿二叉树&#xff09; 文章目录 力扣爆刷第153天之TOP100五连刷&#xff08;相交、翻转、排序链表、螺旋矩阵、锯齿二叉树&#xff09;一、103. 二叉树的锯齿形层序遍历二、92. 反转链表 II三、54…

Python for循环中的引用传递和值传递

先上代码&#xff1a; a [[1],[2],[3]] b [[4,5],[6,7],[7,8]] for i,j in zip(a,b):print(i,j)i [9]#i[0] 8j[:2][1,2]print(i, j) print(a) print(b) 运行的结果&#xff1a; [1] [4, 5] [9] [1, 2] [2] [6, 7] [9] [1, 2] [3] [7, 8] [9] [1, 2] [[1], [2], [3]] [[1…

android常用知识

透明activity样式&#xff1a; android:theme"android:style/Theme.Translucent.NoTitleBar.Fullscreen"这句代码&#xff0c;当你是建的empty activity project时&#xff0c;默认继承的是AppCompat这个类。所以在AndroidMifext.xml文件中用上述代码会导致程序错误&…

ROM 和 RAM

ROM (只读存储器) 和 RAM (随机存取存储器) 是计算机系统中两种不同类型的存储器。它们在定义、用途、工作原理、数据可写性和典型应用方面存在显著差异。 ROM &#xff08;Read-Only Memory&#xff09;(只读存储器) 定义:ROM 是一种只读存储器,用于存储计算机的固件和永久性…

Qt | 简单的使用 QStyle 类(风格也称为样式)

01、前言 者在 pro 文件中已添加了正确的 QT+=widgets 语句 02、基础样式 1、QStyle 类继承自 QObject,该类是一个抽像类。 2、QStyle 类描述了 GUI 的界面外观,Qt 的内置部件使用该类执行几乎所有的绘制,以确保 使这些部件看起来与本地部件完全相同。 3、Qt 内置了一系…

linux段异常信号量

在 Linux 系统中&#xff0c;段异常&#xff08;Segmentation Fault&#xff09;和信号量&#xff08;Semaphore&#xff09;是两个不同的概念&#xff0c;但它们都与进程间通信&#xff08;IPC&#xff09;和错误处理有关。我会分别解释它们&#xff0c;然后解释它们之间可能的…

制造业为什么需要ERP企业管理软件?

如今&#xff0c;传统的制造业管理方式逐渐变得力不从心~库存积压、生产效率低下、供应链混乱…想象一下&#xff0c;如果你的企业仍然依赖于手工记录订单、库存和财务数据&#xff0c;那么每当市场发生变动时&#xff0c;你就需要花费大量的时间和精力去重新调整生产计划、更新…

【深度学习驱动流体力学】VTK创建、处理和可视化流体数据

Visualization Toolkit&#xff08;VTK&#xff09;是一个强大的开源软件系统&#xff0c;用于处理和可视化三维数据。它提供了丰富的工具和算法&#xff0c;可以处理从简单的网格数据到复杂的流体动力学模拟数据等各种类型的数据。本文将完整介绍如何使用 VTK 创建、处理和可视…

Springboot + Mybatis 实现sql打印

参照这个视频&#xff1a;https://www.bilibili.com/video/BV1MS411N7mn/?vd_source90ebeef3261cec486646b6583e9f45f5 实现mybatis对外暴露的接口Interceptor 使用Intercepts接口,这里的写法参照mybatis-plus中的拦截器写法 Intercepts({Signature(type Executor.class, m…

CMake编译proto的方法(custom_target和custom_command)

最近在项目中涉及到在QNX平台上编译CyberRT&#xff0c;其中CyberRT使用到了protobuf&#xff0c;因此&#xff0c;仓库内部有许多proto文件&#xff0c;需要先行将这些proto文件生成对应的.cc和.h文件才能被其他文件使用。 之前一直使用protobuf_generate_cpp来编译proto文件&…

如何在C++中实现延迟删除功能

在软件开发中&#xff0c;缓存是一种常见的优化技术&#xff0c;它允许我们存储数据以供快速访问&#xff0c;从而减少对慢速存储或网络资源的依赖。然而&#xff0c;有时我们可能希望缓存中的某些数据在一段时间后自动过期并被删除&#xff0c;这就是所谓的“延迟删除”功能。…

HTB Freelancer

Freelancer user nmap ➜ htb nmap -A 10.129.221.155 -T 4 Starting Nmap 7.80 ( https://nmap.org ) at 2024-06-02 09:19 CST NSE Timing: About 97.92% done; ETC: 09:24 (0

PostgreSQL源码分析——COPY

导入数据的几种方式 在进行数据导入导出时常会用到copy命令&#xff0c;语法使用可参考下面这篇博文 [Postgres] Bulk Insert and Export Data with csv Files with Postgres copy Command。通常导入数据的方法&#xff0c;可以通过insert的方式&#xff08;insert into t1 va…

BC153 [NOIP2010]数字统计

数字统计 一.题目描述二.输入描述&#xff1a;三.输出描述&#xff1a;四.数字范围五.题目思路六.代码实现 一.题目描述 请统计某个给定范围[L, R]的所有整数中&#xff0c;数字2出现的次数。 比如给定范围[2, 22]&#xff0c;数字2在数2中出现了1次&#xff0c;在数12中出现1次…

如何恢复iPhone iCloud云盘资料删除?给出建议

&#x1f3c6;本文收录于「Bug调优」专栏&#xff0c;主要记录项目实战过程中的Bug之前因后果及提供真实有效的解决方案&#xff0c;希望能够助你一臂之力&#xff0c;帮你早日登顶实现财富自由&#x1f680;&#xff1b;同时&#xff0c;欢迎大家关注&&收藏&&…

【Java】已解决com.mysql.cj.jdbc.exceptions.CommunicationsException异常

文章目录 一、分析问题背景二、可能出错的原因三、错误代码示例四、正确代码示例五、注意事项 已解决com.mysql.cj.jdbc.exceptions.CommunicationsException异常 一、分析问题背景 com.mysql.cj.jdbc.exceptions.CommunicationsException是Java程序在使用MySQL Connector/J与…