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

利用ffmpeg库实现音频Opus编解码

一、编译与环境配置
  1. libopus库集成
    需在编译FFmpeg时添加--enable-libopus参数,编译前需先安装libopus源码并配置动态库路径‌。最新FFmpeg 7.1版本默认支持Opus的浮点运算优化和VBR/CVBR模式‌。

  2. 多平台兼容性
    Opus支持Windows/Linux/macOS平台,编译时需注意不同系统的依赖库路径差异‌。

二、命令行编解码操作
  1. 编码

    # PCM转Opus(48kHz双通道)
    ffmpeg -f s16le -ar 48000 -ac 2 -i input.pcm -c:a libopus -b:a 128k -vbr on output.opus

    -ar:指定输入采样率(支持8k/16k/48k等)‌
    -ac:设置通道数(WebRTC场景强制要求双通道)‌

  2. 解码

    # Opus转PCM并重采样至16kHz单通道
    ffmpeg -i input.opus -ar 16000 -ac 1 -f s16le output.pcm

    支持动态调整输出采样率(8k/16k/44.1k/48k)
    解码后需通过nb_samples获取实际音频帧大小‌

三、代码编解码实现
1、Opus编码代码 
#include <libavcodec/avcodec.h>
#include <libavformat/avformat.h>int decode_opus_to_pcm(const char* input_file, const char* output_file) {AVFormatContext *fmt_ctx = NULL;AVCodecContext *codec_ctx = NULL;const AVCodec *codec = NULL;FILE *pcm_out = fopen(output_file, "wb");// 1. 注册编解码器av_register_all();avcodec_register_all();// 2. 打开输入文件if(avformat_open_input(&fmt_ctx, input_file, NULL, NULL) < 0) {fprintf(stderr, "无法打开输入文件\n");return -1;}// 3. 查找音频流if(avformat_find_stream_info(fmt_ctx, NULL) < 0) {fprintf(stderr, "无法找到流信息\n");return -1;}// 4. 获取Opus解码器codec = avcodec_find_decoder(AV_CODEC_ID_OPUS);if(!codec) {fprintf(stderr, "Opus解码器未找到\n");return -1;}codec_ctx = avcodec_alloc_context3(codec);avcodec_parameters_to_context(codec_ctx, fmt_ctx->streams->codecpar);// 5. 打开解码器if(avcodec_open2(codec_ctx, codec, NULL) < 0) {fprintf(stderr, "无法打开解码器\n");return -1;}AVPacket packet;AVFrame *frame = av_frame_alloc();// 6. 解码循环while(av_read_frame(fmt_ctx, &packet) >= 0) {if(packet.stream_index == 0) {int ret = avcodec_send_packet(codec_ctx, &packet);while(ret >= 0) {ret = avcodec_receive_frame(codec_ctx, frame);if(ret == AVERROR(EAGAIN) || ret == AVERROR_EOF)break;else if(ret < 0) {fprintf(stderr, "解码错误\n");break;}// 7. 输出PCM数据(16-bit小端)fwrite(frame->data, 1, frame->nb_samples * av_get_bytes_per_sample(codec_ctx->sample_fmt), pcm_out);}}av_packet_unref(&packet);}// 8. 清理资源av_frame_free(&frame);avcodec_free_context(&codec_ctx);avformat_close_input(&fmt_ctx);fclose(pcm_out);return 0;
}
2、Opus解码代码 
#include <libavcodec/avcodec.h>
#include <libavutil/opt.h>int encode_pcm_to_opus(const char* input_pcm, const char* output_opus) {const AVCodec *codec = avcodec_find_encoder(AV_CODEC_ID_OPUS);AVCodecContext *codec_ctx = NULL;FILE *opus_out = fopen(output_opus, "wb");FILE *pcm_in = fopen(input_pcm, "rb");// 1. 创建编码器上下文codec_ctx = avcodec_alloc_context3(codec);codec_ctx->bit_rate = 64000;         // 目标码率codec_ctx->sample_fmt = AV_SAMPLE_FMT_S16; // 输入格式codec_ctx->sample_rate = 48000;      // 采样率codec_ctx->channel_layout = AV_CH_LAYOUT_STEREO; // 双声道codec_ctx->channels = 2;// 2. 设置编码参数av_opt_set(codec_ctx->priv_data, "application", "audio", 0); // 音乐优化av_opt_set_int(codec_ctx->priv_data, "compression_level", 10, 0); // 最高质量// 3. 打开编码器if(avcodec_open2(codec_ctx, codec, NULL) < 0) {fprintf(stderr, "无法打开编码器\n");return -1;}AVFrame *frame = av_frame_alloc();frame->nb_samples = codec_ctx->frame_size; // 帧大小(如960 samples@48kHz)frame->format = codec_ctx->sample_fmt;frame->channel_layout = codec_ctx->channel_layout;av_frame_get_buffer(frame, 0);AVPacket *pkt = av_packet_alloc();// 4. 编码循环while(1) {// 读取PCM数据size_t read_size = fread(frame->data, 1, frame->nb_samples * av_get_bytes_per_sample(codec_ctx->sample_fmt), pcm_in);if(read_size <= 0) break;// 发送帧到编码器int ret = avcodec_send_frame(codec_ctx, frame);while(ret >= 0) {ret = avcodec_receive_packet(codec_ctx, pkt);if(ret == AVERROR(EAGAIN) || ret == AVERROR_EOF)break;else if(ret < 0) {fprintf(stderr, "编码错误\n");break;}// 写入输出文件(需添加OGG封装头)fwrite(pkt->data, 1, pkt->size, opus_out);av_packet_unref(pkt);}}// 5. 清理资源av_frame_free(&frame);av_packet_free(&pkt);avcodec_free_context(&codec_ctx);fclose(opus_out);fclose(pcm_in);return 0;
}
3、关键代码说明
  1. 编解码器初始化

    • avcodec_find_decoder/encoder(AV_CODEC_ID_OPUS) 查找编解码器
    • avcodec_alloc_context3() 创建编解码上下文
    • avcodec_open2() 打开编解码器
  2. 数据处理流程

    • 解码‌:av_read_frame() → avcodec_send_packet() → avcodec_receive_frame()
    • 编码‌:avcodec_send_frame() → avcodec_receive_packet()
  3. 参数优化

    // 设置低延迟模式
    av_opt_set(codec_ctx->priv_data, "application", "lowdelay", 0);// 设置VBR模式(0-固定码率,1-可变码率)
    av_opt_set_int(codec_ctx->priv_data, "vbr", 1, 0);// 设置帧持续时间(单位:ms)
    av_opt_set_int(codec_ctx->priv_data, "frame_duration", 20, 0);
    4、编译与运行
    # 编译命令(需链接FFmpeg库)
    gcc opus_example.c -o opus_demo \-lavcodec -lavformat -lavutil -lswresample# 运行解码示例
    ./opus_demo input.opus output.pcm# 运行编码示例(输入需为48kHz双通道s16 PCM)
    ./opus_demo input.pcm output.opus

相关文章:

利用ffmpeg库实现音频Opus编解码

一、编译与环境配置 ‌libopus库集成‌ 需在编译FFmpeg时添加--enable-libopus参数&#xff0c;编译前需先安装libopus源码并配置动态库路径‌。最新FFmpeg 7.1版本默认支持Opus的浮点运算优化和VBR/CVBR模式‌。 ‌多平台兼容性‌ Opus支持Windows/Linux/macOS平台&#xff0…...

SAP FAGLL03 追加并显示描述字段

目录 1、新建一个结构2、操作FAGLPOSX结构3、新建一个BADI 1、新建一个结构 1.1、先在SE11中新建一个结构&#xff1a;ZZADD_FIELDS_FAGL&#xff0c;把我们要显示的描述字段放在这个结构中 2、操作FAGLPOSX结构 2.1、在FAGLPOSX结构中选择Append Structure&#xff0c;把我…...

Linux Vim 寄存器 | 从基础分类到高级应用

注&#xff1a;本文为 “vim 寄存器” 相关文章合辑。 英文引文&#xff0c;机翻未校。 中文引文&#xff0c;略作重排。 未整理去重&#xff0c;如有内容异常&#xff0c;请看原文。 Registers 寄存器 Learning Vim registers is like learning algebra for the first ti…...

Ubuntu版免翻墙搭建BatteryHistorian

摘要 昨天安装了一个翻墙版本的很不好用&#xff0c;主要是网络不稳定&#xff0c;故于是换了一个免翻墙的docker镜像。但是发现还是很难用。又安装了一个window版本的免翻墙的BatteryHistorian。明天再分享下Windows的免翻墙的BatteryHistorian步骤。 安装好Docker了就直接d…...

Django Rest Framework 创建纯净版Django项目部署DRF

描述创建纯净版的Django项目和 Django Rest Framework 环境的部署 一、创建Django项目 1. 环境说明 操作系统 Windows11python版本 3.9.13Django版本 V4.2.202. 操作步骤(在Pycharm中操作) 创建Python项目drfStudy、虚拟环境 ​虚拟环境中安装 jdangopip install django==4.…...

深度洞察:DeepSeek 驱动金融行业智能化转型变革

该文章为软件测评&#xff0c;不是广告&#xff01;&#xff01;&#xff01;&#xff01; 目录 一.金融行业的智能化转型浪潮​ 二.DeepSeek的核心技术剖析 1.DeepSeek 模型的金融智慧​ 2.实时联网搜索&#xff1a;把握金融市场脉搏​ 3.RAG 能力&#xff1a;铸就精准金…...

面试题精选《剑指Offer》:JVM类加载机制与Spring设计哲学深度剖析-大厂必考

一、JVM类加载核心机制 &#x1f525; 问题5&#xff1a;类从编译到执行的全链路过程 完整生命周期流程图 关键技术拆解 编译阶段 查看字节码指令&#xff1a;javap -v Robot.class 常量池结构解析&#xff08;CONSTANT_Class_info等&#xff09; 类加载阶段 // 手动加载…...

【CXX-Qt】2.1.1 为 WebAssembly 构建

CXX-Qt 及其编写的应用程序可以编译为 WebAssembly&#xff0c;但存在一些限制。以下是关于如何为 WASM 目标构建的详细说明。 你需要安装 Qt for WebAssembly。下一篇将展示已测试的版本。 此外&#xff0c;如果尚未完成&#xff0c;请从此处克隆 emsdk git 仓库。 使用正确…...

AUTOSAR Communication Services - COM:(二)COM的常见API用法整理

备注&#xff1a;COM-API常用用法整理&#xff0c;持续更新 一、用户I-PDU发送回调中&#xff0c;指定发送对应DBC的信号值 boolean Rte_COMIPduCallout_signal(PduIdType id, PduInfoType *ptr) {static uint8 ucCheckSum 0;// Calculate checksumif (ucCheckSum > 15)u…...

掌握些许 IPv6 要点,windows 远程桌面安全便利两相宜!

掌握这些要点&#xff0c;Windows 远程桌面安全便利两相宜&#xff01; 在日常办公中&#xff0c;许多人会用到 Windows 系统的远程桌面功能。但在实际使用时&#xff0c;会遇到内网计算机难以通过运营商的动态 ip 与多层 NAT 向互联网暴露端口的技术问题&#xff0c;和计算机…...

Spring Boot整合Apache BookKeeper教程

精心整理了最新的面试资料和简历模板&#xff0c;有需要的可以自行获取 点击前往百度网盘获取 点击前往夸克网盘获取 Spring Boot整合Apache BookKeeper教程 1. 简介 Apache BookKeeper 是一个高性能、持久化的分布式日志存储系统&#xff0c;适用于需要强一致性和高吞吐量的…...

【Linux进程】——进程的程序地址空间

目录 前言 1.程序地址空间 1.1区域划分 1.2程序地址空间的本质 1.3程序地址空间分配原则 2.数据寻找 2.1补充&#xff1a;进程挂起 结语 前言 在Linux系统的神秘世界里&#xff0c;进程就像是一个个小工匠&#xff0c;各自忙碌地完成着不同的任务。你是否想过&#xff…...

边缘云原生操作系统的设计与思考

资料来源&#xff1a;火山引擎-开发者社区 边缘云行业现状和发展历程 从 06 年 AWS 推出 EC2 、S3 到今天已经过去了 18 年&#xff0c;云计算早已不是一个新鲜词汇&#xff0c;从当前业务来看&#xff0c;我们能看到云计算从中心到中心边缘的发展趋势&#xff0c;为什么会有 这…...

Jenkins muti-configuration-project 中调用pipeline project

Jenkins muti-configuration-project 中调用pipeline project 解决方案示例练习1. 多配置项目设置&#xff1a;2. 触发器配置&#xff1a;3. Pipeline 项目 Jenkinsfile&#xff1a; 解决方案 创建多配置项目&#xff1a; 在 Jenkins 中创建一个新的多配置项目。在“配置矩阵…...

Gdiplus(也就是GDI+)使用

篇一&#xff1a; Gdiplus(也就是GDI)使用步骤&#xff1a; 1.包括相应的头文件及引入相应的lib #include <GdiPlus.h> #pragma comment(lib, "gdiplus.lib") using namespace Gdiplus;//如果没有using namespace Gdiplus;就需要添加“命名空间作用域符” …...

AI学习——卷积神经网络(CNN)入门

作为人类&#xff0c;我们天生擅长“看”东西&#xff1a;一眼就能认出猫狗、分辨红绿灯、读懂朋友的表情……但计算机的“眼睛”最初是一片空白。直到卷积神经网络&#xff08;CNN&#xff09;​的出现&#xff0c;计算机才真正开始理解图像。今天&#xff0c;我们就用最通俗的…...

双指针算法-day14(分组循环)

1.最长奇偶子数组 题目 解析 分组循环模板&#xff1a; 简单来说&#xff1a; 第一步&#xff1a;指针遍历找到满足条件的开头下标&#xff0c;并用 start i 记录开头&#xff1b;第二步&#xff1a;指针不断右移寻找满足条件的最长子数组&#xff1b;第三步&#xff1a;更新…...

Linux基础开发工具--gdb的使用

目录 安装准备&#xff1a; 1. 背景 2. 开始使用 3. 做一个Linux第一个小程序&#xff0d;进度条 安装准备&#xff1a; 对于gdb的学习使用&#xff0c;为了方便大家学习&#xff0c;我建议大家先安装一个cgdb进行学习&#xff0c;这样方便观察操作与学习gdb。 用以下…...

垃圾回收算法(Garbage Collection)深度解析

垃圾回收算法&#xff08;Garbage Collection&#xff09;深度解析 核心思想 垃圾回收核心目标&#xff1a; 定位存活对象 、回收死亡对象可达性分析算法、空间复用优化 生死判定原理 可达性分析法&#xff08;Reachability Analysis&#xff09; 通过GC Roots&#xff08…...

基于Matlab实现语音识别算法(源码+数据)

语音识别技术是现代信息技术中的一个重要领域&#xff0c;特别是在人机交互、智能设备、智能家居、自动驾驶等多个领域有着广泛应用。MATLAB作为一种强大的数值计算和数据可视化环境&#xff0c;因其易用性和丰富的库支持&#xff0c;常被用来实现复杂的算法&#xff0c;包括语…...

RabbitMQ的高级特性介绍(一)

消息确认机制 ⽣产者发送消息之后, 到达消费端之后, 可能会有以下情况: a. 消息处理成功 b. 消息处理异常 RabbitMQ向消费者发送消息之后, 就会把这条消息删掉, 那么第二种情况, 就会造成消息丢失。 那么如何确保消费端已经成功接收了, 并正确处理了呢? 为了保证消息从队列…...

QML开发入门1--安装QT6.8和新建第一个QtQuickApplication

1.下载在线安装工具 qt-online-installer-windows-x64-4.8.1.exe 2.安装 注&#xff1a;可能官网qt安装很慢。需要使用国内镜像源。推荐阿里镜像 qt-online-installer-windows-x64-4.8.1.exe --mirror https://mirrors.aliyun.com/qt3.配置QT关键配置 3.1 无法编译 注&#…...

指令系统2(Load/Store 指令)

一. Load/Store 指令 1. 前变址 前变址指令是在读取或存储数据时&#xff0c;先根据基址寄存器&#xff08;Rn&#xff09;与偏移量&#xff08;offset&#xff09;计算出有效地址&#xff0c;再进行数据操作。相关指令及示例如下&#xff1a; LDR R0, [R1, #4]&#xff1a;从…...

【实战案例】用STAR+3W模型拆解电商支付系统设计文档

各位开发者朋友&#xff0c;上次分享了结构化写作的黄金公式后&#xff0c;很多同学反馈需要更具象的落地方法。今天通过真实电商支付系统案例&#xff0c;手把手教你用STAR3W模型写出可执行的设计文档&#xff01; 结构化写作的「黄金公式」 STAR原则 3W模型 Situation&…...

C#使用SnsPictureBox.dll绘制点,线段、圆、折线、多边形、测量尺等多种图形。

CSDN下载地址&#xff1a;https://download.csdn.net/download/sns1991sns/87726867 gitee下载地址:https://gitee.com/linsns/SnsPictrueBox 支持2种绘制方式&#xff1a;响应式和等待式。 一、使用响应式绘制图形 1、在窗口构造函数里添加绘制图形的完成响应函数 public…...

如何让节卡机器人精准对点?

如何让节卡机器人精准对点&#xff1f; JAKA Zu 软件主界面主要由功能栏、开关栏、菜单栏构成。 菜单栏&#xff1a;控制柜管理&#xff0c;机器人管理与软件管理组成。主要功能为对控制柜关机、APP 设置、机器人本体设 置、控制柜设置、连接机器人和机器人显示等功能。 开关…...

文转语音好用的平台

一、国外平台推荐 第一梯队&#xff1a;全球头部服务 Amazon Polly&#xff08;AWS&#xff09; 特点&#xff1a;支持 70语言/方言&#xff0c;提供神经语音&#xff08;NTTS&#xff09;和标准语音&#xff0c;可克隆声音&#xff08;Voice Designer&#xff09;。平台&…...

常见JavaScript页面部分内容显示/隐藏设置总结

项目中经常遇到通过js判断对页面中某一部分进行显示/隐藏设置的场景&#xff0c;经常使用的是display&#xff0c;有时也会使用visibility、opacity&#xff0c;为此&#xff0c;特意查询了相关内容&#xff0c;对其进行了一下汇总记录下&#xff1a; 除了 display&#xff0c…...

在 Spring Boot 中调用 AnythingLLM 的发消息接口

整体逻辑: 自建系统的web UI界面调用接口: 1.SpringBoot接口&#xff1a;/anything/chatMessageAnything 2.调用anythingLLM - 调用知识库deepseek r1 . Windows Installation ~ AnythingLLMhttps://docs.anythingllm.com/installation-desktop/windows http://localhost:3…...

TextView、AppCompatTextView和MaterialTextView该用哪一个?Android UI 组件发展史与演进对照表

在 Android 开发中&#xff0c;UI 组件一直在不断演进&#xff0c;从最初的原生组件&#xff0c;到 Support Library&#xff08;AppCompat 兼容库&#xff09;&#xff0c;再到如今的 Material Design 组件。这篇文章将梳理 Android UI 组件的发展历史&#xff0c;并提供详细的…...