FFmpeg PCM编码为AAC

使用FFmpeg库把PCM文件编码为AAC文件,FFmpeg版本为4.4.2-0

代码如下:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <libavcodec/avcodec.h>
#include <libavformat/avformat.h>
#include <libavutil/opt.h>
#include <libavutil/mem.h>
#include <libavutil/samplefmt.h>
#include <libswresample/swresample.h>int main(int argc, char *argv[])
{int ret = -1;int64_t pts = 0;const char *input_file = argv[1];const char *output_file = argv[2];FILE *input_pcm = NULL;AVFormatContext *format_context = NULL;AVCodecContext *codec_context = NULL;AVStream *stream = NULL;AVCodec *codec = NULL;AVFrame *frame = NULL;AVPacket *packet = NULL;struct SwrContext *swr_ctx = NULL;enum AVSampleFormat in_sample_fmt = AV_SAMPLE_FMT_S16; // 输入pcm文件的采样格式int in_sample_rate = 44100;                            // 输入pcm文件的采样率uint64_t in_channel_layout = AV_CH_LAYOUT_STEREO;      // 输入pcm文件的声道布局if (argc < 3){fprintf(stderr, "Usage: %s <input file> <output file>\n", argv[0]);return -1;}input_pcm = fopen(input_file, "rb");if (!input_pcm){fprintf(stderr, "Could not open input file %s\n", input_file);goto end;}// 分配输出格式上下文avformat_alloc_output_context2(&format_context, NULL, NULL, output_file);if (!format_context){fprintf(stderr, "Could not allocate output format context\n");goto end;}// 查找编码器, 这里使用ffmpeg内置的aac编码器,// 如果需要使用其他编码器, 可以使用 avcodec_find_encoder_by_name 按名称查找codec = avcodec_find_encoder(AV_CODEC_ID_AAC);if (!codec){fprintf(stderr, "Codec not found\n");goto end;}// 创建新的音频流stream = avformat_new_stream(format_context, NULL);if (!stream){fprintf(stderr, "Could not allocate stream\n");goto end;}// 分配编码器上下文codec_context = avcodec_alloc_context3(codec);if (!codec_context){fprintf(stderr, "Could not allocate codec context\n");goto end;}/* 设置编码器参数 */// 编码器IDcodec_context->codec_id = AV_CODEC_ID_AAC;// 媒体类型codec_context->codec_type = AVMEDIA_TYPE_AUDIO;// 设置编码器接收的输入音频样本的采样格式,这里设置为内置aac编码器支持的采样格式,// 不同的编码器支持的采样格式可能不同。codec_context->sample_fmt = AV_SAMPLE_FMT_FLTP;// 设置采样率codec_context->sample_rate = in_sample_rate;// 设置声道布局codec_context->channel_layout = in_channel_layout;// 设置声道数codec_context->channels = av_get_channel_layout_nb_channels(codec_context->channel_layout);// 设置比特率codec_context->bit_rate = 128000;// 设置AAC的profilecodec_context->profile = FF_PROFILE_AAC_LOW;// 打开编码器if (avcodec_open2(codec_context, codec, NULL) < 0){fprintf(stderr, "Could not open codec\n");goto end;}// 将编码器参数复制到流int result = avcodec_parameters_from_context(stream->codecpar, codec_context);if (result < 0){fprintf(stderr, "Could not copy codec parameters\n");goto end;}// 打开输出文件if (!(format_context->oformat->flags & AVFMT_NOFILE)) // 检查输出格式是否需要文件存储{result = avio_open(&format_context->pb, output_file, AVIO_FLAG_WRITE);if (result < 0){fprintf(stderr, "Could not open output file %s\n", output_file);goto end;}}// 写文件头result = avformat_write_header(format_context, NULL);if (result < 0){fprintf(stderr, "Error occurred when opening output file\n");goto end;}frame = av_frame_alloc();packet = av_packet_alloc();if (!frame || !packet){fprintf(stderr, "Could not allocate frame or packet\n");goto end;}// 设置帧参数frame->nb_samples = codec_context->frame_size; // 一帧音频数据的采样个数,AAC编码器的frame_size默认为1024frame->format = codec_context->sample_fmt;frame->channel_layout = codec_context->channel_layout;// 分配帧数据缓冲区result = av_frame_get_buffer(frame, 0);if (result < 0){fprintf(stderr, "Could not allocate audio data buffers\n");goto end;}// 初始化重采样上下文swr_ctx = swr_alloc_set_opts(NULL,codec_context->channel_layout, codec_context->sample_fmt, codec_context->sample_rate,codec_context->channel_layout, in_sample_fmt, codec_context->sample_rate,0, NULL);if (!swr_ctx || swr_init(swr_ctx) < 0){fprintf(stderr, "Could not allocate resampler context\n");goto end;}// 计算每帧音频数据的大小 = 采样个数 * 采样格式大小 * 声道数int fsize = frame->nb_samples * av_get_bytes_per_sample(in_sample_fmt) * codec_context->channels;// 分配缓冲区uint8_t *input_buffer = (uint8_t *)av_malloc(fsize);if (!input_buffer){fprintf(stderr, "Could not allocate input buffer\n");goto end;}// 编码循环while (1){// 读取PCM数据到缓冲区result = fread(input_buffer, 1, fsize, input_pcm);if (result <= 0){break;}// 处理不足一帧的情况,不足一帧的数据用0填充if (result < fsize){memset(input_buffer + result, 0, fsize - result);}// 重采样result = swr_convert(swr_ctx, frame->data, frame->nb_samples,(const uint8_t **)&input_buffer, frame->nb_samples);if (result < 0){fprintf(stderr, "Error while converting\n");goto end;}frame->pts = pts;pts += frame->nb_samples; // 计算下一帧的pts,每个帧的时间戳应该是前一个帧的时间戳加上该帧的样本数// 发送帧到编码器result = avcodec_send_frame(codec_context, frame);if (result < 0){fprintf(stderr, "Error sending frame to codec\n");goto end;}// 接收编码后的数据包while (result >= 0){result = avcodec_receive_packet(codec_context, packet);if (result == AVERROR(EAGAIN) || result == AVERROR_EOF){break;}else if (result < 0){fprintf(stderr, "avcodec_receive_packet failed (errmsg '%s')\n", av_err2str(result));goto end;}packet->stream_index = stream->index;// 将时间戳从编码器时间基转换到流时间基av_packet_rescale_ts(packet, codec_context->time_base, stream->time_base);// 写数据包到输出文件result = av_interleaved_write_frame(format_context, packet);if (result < 0){fprintf(stderr, "Error writing audio packet\n");av_packet_unref(packet);goto end;}av_packet_unref(packet);}}av_free(input_buffer);// 发送NULL帧到编码器,刷新编码器内部缓冲区result = avcodec_send_frame(codec_context, NULL);while (result >= 0){result = avcodec_receive_packet(codec_context, packet);if (result == AVERROR_EOF){break;}else if (result < 0){fprintf(stderr, "avcodec_receive_packet failed (errmsg '%s')\n", av_err2str(result));goto end;}packet->stream_index = stream->index;av_packet_rescale_ts(packet, codec_context->time_base, stream->time_base);result = av_interleaved_write_frame(format_context, packet);if (result < 0){fprintf(stderr, "Error writing audio packet\n");av_packet_unref(packet);goto end;}av_packet_unref(packet);}// 写文件尾av_write_trailer(format_context);ret = 0;end:if (frame)av_frame_free(&frame);if (packet)av_packet_free(&packet);if (codec_context)avcodec_free_context(&codec_context);if (format_context)avformat_free_context(format_context);if (swr_ctx)swr_free(&swr_ctx);if (input_pcm)fclose(input_pcm);return ret;
}

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

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

相关文章

Codes 重新定义 SaaS 模式的研发项目管理平台开源版 4.5.5 发布

一&#xff1a;简介 Codes 重新定义 SaaS 模式 云端认证 程序及数据本地安装 不限功能 30 人免费 Codes 是一个 高效、简洁、轻量的一站式研发项目管理平台。包含需求管理&#xff0c;任务管理&#xff0c;测试管理&#xff0c;缺陷管理&#xff0c;自动化测试&#xff0…

海外短剧的未来展望:创新与发展的方向

随着全球化的加速和互联网技术的飞速发展&#xff0c;海外短剧作为一种新兴的娱乐形式&#xff0c;正逐渐赢得广大观众的喜爱。在这个充满变革的时代&#xff0c;海外短剧面临着前所未有的机遇与挑战。本文将探讨海外短剧未来的创新与发展方向。 一、内容创新&#xff1a;打破传…

Oracle操作扩可变字符长度交易影响分析-较小

使用AI帮助学习知识 以下知识来至AI oracle 一张大表&#xff0c;对可变字符串长度从10扩到20位&#xff0c;oracle底层存储是否会发生变化&#xff0c;先锁表&#xff0c;更新表字典信息&#xff0c;然后会不会重新整理表&#xff0c;在有交易的情况下导致大量交易失效&#…

日常开发坑记录

hutool工具类转换,anInt可能为负数(队列散列需求遇到)long l = RandomUtil.randomLong(0, 9999999999L);Integer anInt = Convert.toInt(l);System

wandb上传整个项目代码

wandb.run.log_code上传整个项目代码 wandb一般只能保存训练代码文件 用了这个函数并且指定文件格式&#xff0c;就能保存指定文件当git来用了 import wandbwandb.init(project"latent-diffusion") wandb.run.log_code(".", include_fnlambda path: path.e…

docker --network host,仍然无法使用127.0.0.1连接数据库,改用宿主机ip后可以连接

有两个docker容器&#xff0c;一个是mysql&#xff0c;一个是java服务。两个都是host网络模式。 宿主机上可以连mysql&#xff0c;java服务也可以通过wget检测能连上mysql端口。 外部网络也能连上mysql。只有这个java服务无法连接mysql&#xff0c;java服务是用是127.0.0.1&…

网络ip地址冲突会出现什么情况

在现代数字化社会中&#xff0c;网络IP地址扮演着至关重要的角色&#xff0c;它是设备在网络中唯一识别的标识。然而&#xff0c;当网络中出现IP地址冲突时&#xff0c;一系列问题便会随之而来。那么&#xff0c;网络ip地址冲突会出现什么情况呢&#xff1f;下面一起来跟虎观代…

k8s-部署对象存储minio

环境信息 minio版本 :最新 k8s 版本1.22 使用nfs作为共享存储 一.单节点安装包部署 脚本部署&#xff0c;一键部署&#xff0c;单节点应用于数据量小&#xff0c;一些缓存存储&#xff0c;比如gitlab-runner的产物数据&#xff0c;maven的打包依赖数据 #!/bin/bash# 步骤…

如何进行辐射抗扰度磁场测试?

一、为什么要进行闭环磁场测试&#xff1f; 辐射抗扰度测试中进行闭环磁场测试是为了评估设备在外部磁场影响下的性能表现。外部磁场可能来自各种源头&#xff0c;例如电力线、电动机、变压器等&#xff0c;这些磁场可能干扰设备的正常工作。闭环磁场测试通过模拟设备在实际工…

虾皮Lazada流量“滑铁卢”?自养号测评补单让你轻松翻盘!

当作为虾皮&#xff08;Shopee&#xff09;或Lazada平台的卖家时&#xff0c;密切关注流量数据是至关重要的。如果观察到店铺的流量出现下滑趋势&#xff0c;首要任务便是深入探究流量减少的根本原因。在明确了导致流量下滑的关键因素后&#xff0c;卖家便能更有针对性地采取措…

Java学习20——Map接口

目录 一.Map&#xff1a; 1.基本介绍&#xff1a; 2.Map常用方法&#xff1a; 3.Map的遍历方法&#xff1a; 4.HashMap: 1.基本介绍&#xff1a; 2.HashMap底层扩容机制&#xff1a; 5.Hashtable&#xff1a; 1.基本介绍&#xff1a; 2.HashMap和Hashtable的对比&…

JSON字符串

在Python中&#xff0c;你可以使用内置的json模块将JSON字符串解析为Python对象&#xff08;如字典、列表等&#xff09;。以下是一个简单的示例&#xff0c;说明如何将JSON字符串解析为Python字典&#xff1a; python复制代码 import json # JSON字符串 json_string {"…

Python环境变量设置Win7:深度解析与实用指南

Python环境变量设置Win7&#xff1a;深度解析与实用指南 在Windows 7操作系统中设置Python环境变量&#xff0c;是每位Python开发者都需要掌握的一项基本技能。它不仅能够使我们在命令行中直接运行Python脚本&#xff0c;还能够确保Python相关的库和工具能够被系统正确识别和使…

Image Search-这是你的图像搜索

Image Search-这是你的图像搜索 什么是图像搜索图像搜索开通图像搜索商品图片搜索图片搜索图片新增批量操作OSS-创建bucket上传文件创建increment.meta并上传元信息导出 体验感受 什么是图像搜索 在接触一个新的产品时&#xff0c;我们首先要知道这款产品是什么&#xff1f;那…

ARM IHI0069F GIC architecture specification (8)

3.2中断旁路支持 CPU interface可以支持中断信号旁路&#xff0c;使得当接口发出的中断信号被禁用时&#xff0c;传统中断信号被传递到PE上的中断请求输入&#xff0c;从而绕过GIC功能。 是否支持旁路由实际设计决定。 用于确定是否使用GICv3 FIQ和IRQ输出或旁路信号的控制取决…

英伟达在2024台北电脑展上推出NIM云原生微服务

&#x1f989; AI新闻 &#x1f680; 英伟达推出NIM云原生微服务&#xff0c;助力生成式AI发展 摘要&#xff1a;在2024台北电脑展上&#xff0c;黄仁勋介绍了生成式人工智能将推动软件全栈重塑&#xff0c;并展示了Nvidia Inference Microservices&#xff08;NIM&#xff0…

《2024快手未成年人保护报告》发布:八大功能保护未成年人隐私信息

5月31日&#xff0c;快手发布《2024快手未成年人保护报告》&#xff08;下文简称《报告》&#xff09;。《报告》显示&#xff0c;快手在2023年持续升级未成年人保护机制&#xff0c;在不断提升平台保障能力和未成年人使用体验的同时&#xff0c;针对未成年人隐私信息保护持续优…

区间预测 | Matlab实现GRU-Attention-KDE核密度估计多置信区间多变量回归区间预测

区间预测 | Matlab实现GRU-Attention-KDE核密度估计多置信区间多变量回归区间预测 目录 区间预测 | Matlab实现GRU-Attention-KDE核密度估计多置信区间多变量回归区间预测效果一览基本介绍程序设计参考资料 效果一览 基本介绍 1.Matlab实现GRU-Attention-KDE门控循环单元注意力…

【JS实战02】轮播图

一&#xff1a;HTML页面结构 1 整体外观 2 HTML结构以及CSS样式 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0">…

js之对象、内置对象、获取和操作DOM对象以及相关年会抽奖案例

这里写目录标题 一级目录二级目录三级目录 基础语法04一、对象1.定义2.特点 二、对象的使用1.声明2.由属性和方法组成3.属性4.增删改查5.方法 三、遍历对象四、内置对象1.介绍2.作用3.方法注意&#xff1a;生成任意范围的随机数 五、声明变量注意WebAPIs01一、Web API 基本认知…