小智机器人关键函数解析,Application::OutputAudio()处理音频数据的输出的函数
以下是对 Application::OutputAudio() 函数的详细解释:
源码:
void Application::OutputAudio() { // 扬声器的输出auto now = std::chrono::steady_clock::now();auto codec = Board::GetInstance().GetAudioCodec();const int max_silence_seconds = 10;std::unique_lock<std::mutex> lock(mutex_);if (audio_decode_queue_.empty()) {// Disable the output if there is no audio data for a long timeif (device_state_ == kDeviceStateIdle) {auto duration = std::chrono::duration_cast<std::chrono::seconds>(now - last_output_time_).count();if (duration > max_silence_seconds) {codec->EnableOutput(false);}}return;}if (device_state_ == kDeviceStateListening) {audio_decode_queue_.clear();return;}last_output_time_ = now;auto opus = std::move(audio_decode_queue_.front());audio_decode_queue_.pop_front();lock.unlock();background_task_->Schedule([this, codec, opus = std::move(opus)]() mutable {if (aborted_) {return;}std::vector<int16_t> pcm;if (!opus_decoder_->Decode(std::move(opus), pcm)) {return;}// Resample if the sample rate is differentif (opus_decode_sample_rate_ != codec->output_sample_rate()) {int target_size = output_resampler_.GetOutputSamples(pcm.size());std::vector<int16_t> resampled(target_size);output_resampler_.Process(pcm.data(), pcm.size(), resampled.data());pcm = std::move(resampled);}codec->OutputData(pcm);});
}
函数概述
Application::OutputAudio() 函数主要负责处理音频数据的输出,通过扬声器播放音频。它会检查音频解码队列的状态,根据不同情况决定是否输出音频、是否禁用输出设备,并且会对音频数据进行解码和重采样处理,最后将处理后的音频数据发送到音频编解码器进行输出。
代码详细解释
1. 时间获取和编解码器获取
auto now = std::chrono::steady_clock::now();
auto codec = Board::GetInstance().GetAudioCodec();
const int max_silence_seconds = 10;
now:获取当前时间,用于后续计算音频数据的空闲时长。codec:通过Board::GetInstance().GetAudioCodec()获取音频编解码器的实例,用于音频数据的输出。max_silence_seconds:定义了最大允许的静音时长,单位为秒。
2. 加锁并检查音频解码队列
std::unique_lock<std::mutex> lock(mutex_);
if (audio_decode_queue_.empty()) {if (device_state_ == kDeviceStateIdle) {auto duration = std::chrono::duration_cast<std::chrono::seconds>(now - last_output_time_).count();if (duration > max_silence_seconds) {codec->EnableOutput(false);}}return;
}
std::unique_lock<std::mutex> lock(mutex_):对互斥锁mutex_加锁,确保在访问audio_decode_queue_时线程安全。audio_decode_queue_.empty():检查音频解码队列是否为空。- 如果队列为空且设备状态为
kDeviceStateIdle,计算从上次输出音频到现在的时长duration。 - 如果
duration超过max_silence_seconds,调用codec->EnableOutput(false)禁用音频输出设备。 - 无论是否禁用输出设备,只要队列为空,函数直接返回。
- 如果队列为空且设备状态为
3. 检查设备状态
if (device_state_ == kDeviceStateListening) {audio_decode_queue_.clear();return;
}
- 如果设备状态为
kDeviceStateListening,清空音频解码队列并返回,不进行音频输出。
4. 更新最后输出时间并取出音频数据
last_output_time_ = now;
auto opus = std::move(audio_decode_queue_.front());
audio_decode_queue_.pop_front();
lock.unlock();
last_output_time_ = now:更新最后输出音频的时间。auto opus = std::move(audio_decode_queue_.front()):将队列头部的音频数据(OPUS 格式)移动到opus变量中。audio_decode_queue_.pop_front():从队列中移除头部元素。lock.unlock():解锁互斥锁,允许其他线程访问audio_decode_queue_。
5. 调度后台任务处理音频数据
background_task_->Schedule([this, codec, opus = std::move(opus)]() mutable {if (aborted_) {return;}std::vector<int16_t> pcm;if (!opus_decoder_->Decode(std::move(opus), pcm)) {return;}if (opus_decode_sample_rate_ != codec->output_sample_rate()) {int target_size = output_resampler_.GetOutputSamples(pcm.size());std::vector<int16_t> resampled(target_size);output_resampler_.Process(pcm.data(), pcm.size(), resampled.data());pcm = std::move(resampled);}codec->OutputData(pcm);
});
background_task_->Schedule(...):将一个 lambda 函数调度到后台任务中执行。if (aborted_):检查是否已终止操作,如果是则直接返回。opus_decoder_->Decode(std::move(opus), pcm):使用opus_decoder_对 OPUS 格式的音频数据进行解码,解码结果存储在pcm向量中。如果解码失败,函数返回。if (opus_decode_sample_rate_ != codec->output_sample_rate()):检查解码后的音频采样率是否与音频编解码器的输出采样率一致。如果不一致,进行重采样处理:output_resampler_.GetOutputSamples(pcm.size()):获取重采样后的样本数量。output_resampler_.Process(pcm.data(), pcm.size(), resampled.data()):执行重采样操作。pcm = std::move(resampled):将重采样后的音频数据移动到pcm向量中。
codec->OutputData(pcm):将处理后的 PCM 音频数据发送到音频编解码器进行输出。
代码优化建议
- 错误处理:在
codec->OutputData(pcm)调用后,可以添加错误处理逻辑,以处理可能的输出错误。 - 日志记录:在关键步骤添加日志记录,方便调试和监控程序运行状态。
- 资源管理:确保
opus_decoder_和output_resampler_在使用完毕后正确释放资源。
相关文章:
小智机器人关键函数解析,Application::OutputAudio()处理音频数据的输出的函数
以下是对 Application::OutputAudio() 函数的详细解释: 源码: void Application::OutputAudio() { // 扬声器的输出auto now std::chrono::steady_clock::now();auto codec Board::GetInstance().GetAudioCodec();const int max_silence_seconds 10;…...
玛卡巴卡的k8s知识点问答题(五)
17. Init 类型容器有什么特点,主要用途? 特点: 启动顺序:Init 容器在普通容器启动之前运行,必须先完成所有 Init 容器后,Pod 的主容器才会启动。 顺序执行:如果定义了多个 Init 容器ÿ…...
3.27学习总结 爬虫+二维数组+Object类常用方法
高精度: 一个很大的整数,以字符串的形式进行接收,并将每一位数存储在数组内,例如100,即存储为[1][0][0]。 p2437蜜蜂路线 每一个的路线数前两个数的路线数相加。 #include <stdio.h> int a[1005][1005]; int …...
kafka零拷贝技术的底层实现
什么是 Sendfile? sendfile 是一种操作系统提供的系统调用(system call),用于在两个文件描述符(file descriptor)之间高效传输数据。它最初由 Linux 内核引入(从 2.1 版本开始)&…...
MFC中CMap类的用法和原理
1、CMap 的原理 CMap 是一个基于哈希表的映射类,它将唯一键映射到对应的值。其内部实现依赖于哈希算法,通过哈希函数将键转换为哈希值,然后将哈希值映射到哈希表中的某个位置。如果多个键的哈希值相同(即哈希冲突)&am…...
elementplus的el-tabs路由式
在使用 Element Plus 的 el-tabs 组件,实现路由式的切换(即点击标签页来切换不同的路由页面)。下面是一个基于 Vue 3 和 Element Plus 实现路由式 el-tabs 的基本步骤和示例。 步骤 1: 安装必要的库 在vue3项目安装 Vue Router 和 Element …...
数据结构初阶:单链表
序言: 本篇博客主要介绍单链表的基本概念,包括如何定义和初始化单链表,以及如何进行数据的插入,删除和销毁等操作。 1.单链表 1.1 概念与结构 概念:链表是一种非顺序的存储结构,数据元素的逻辑顺序是通过…...
北斗导航 | 改进伪距残差矢量的接收机自主完好性监测算法原理,公式,应用,RAIM算法研究综述,matlab代码
改进伪距残差矢量的接收机自主完好性监测算法研究 摘要 接收机自主完好性监测(RAIM)是保障全球卫星导航系统(GNSS)可靠性的核心技术。本文针对传统伪距残差矢量法在微小故障检测和多故障隔离中的不足,提出一种融合加权奇偶空间与动态阈值调整的改进算法。通过理论推导验证…...
RabbitMQ高级特性--TTL和死信队列
目录 1.TTL 1.1设置消息的TTL 1.1.1配置交换机&队列 1.1.2发送消息 1.1.3运行程序观察结果 1.2设置队列的TTL 1.2.1配置队列和交换机的绑定关系 1.2.2发送消息 1.2.3运行程序观察结果 1.3两者区别 2.死信队列 2.1 声名队列和交换机 2.2正常队列绑定死信交换机 …...
Java后端开发: 如何安装搭建Java开发环境《安装JDK》和 检测JDK版本
文章目录 一、JDK的安装1、 打开 Oracle 官方网址2、点击产品 二、检测JDK是否安装成功以及JDK版本的查看1. 打开命令行窗口检测是否安装成功查看 JDK 版本 一、JDK的安装 1、 打开 Oracle 官方网址 Oracle官网地址:https://www.oracle.com/cn/ 2、点击产品 打开下载的JDK文件…...
LabVIEW液压控制系统开发要点
液压控制系统开发需兼顾高实时性、强抗干扰性和安全性,尤其在重工业场景中,毫秒级响应延迟或数据异常都可能导致设备损坏。本文以某钢厂液压升降平台项目为例,从硬件选型、控制算法、安全机制三方面,详解LabVIEW开发中的关键问题与…...
鸿蒙Flutter实战:18-组合而非替换,现有插件快速鸿蒙化
引言 在对插件鸿蒙化时,除了往期文章现有Flutter项目支持鸿蒙II中讲到的使用 dependency_overrides 来配置鸿蒙适配库的两种方式以外,如果三方插件本身使用了联合插件的形式,也可以通过下面这种方式来添加鸿蒙平台的实现: depen…...
Qt之Service开发
一、概述 基于Qt的用于开发系统服务(守护进程)和后台服务,有以下几个优秀的开源 QtService 框架和库。 1. QtService (官方解决方案) GitHub: https://github.com/qtproject/qt-solutions/tree/master/qtservice 特点: 官方提供的服务框架 支持 Windows 服务和 Linux 守护…...
MFC添加免费版大漠3.1233
先创建一个MFC工程, 添加dm.dll 方法一:通过类向导-添加类-类型库中的MFC类-文件,选择dm.dll,如果没有"添加类型库中的MFC类"选项就用方法二添加 方法二:添加-新建项-MFC-Active或TypeLib-实现接口位置选…...
vue 图片放大到全局
背景: 在vue项目中,el-image组件图片组件用于展示图片,组件自带的属性preview-teleported,设置为true可以控制图片放大到全局 实现效果: 核心代码: //图片地址:BASEUrl /file/ item.file //这…...
人工智能入门(1)
人工智能导引 文章目录 人工智能导引artifiicial intelligence由图灵测试出发的六个领域贝叶斯方法分析成为大多数AI系统中不确定推理的现代方法基础 研究方法 机器学习计算机利用已经有的数据样本,得出某种规律模型,并利用模型预测未来的一种方法 回归算…...
Python爬虫:Feapder 的详细使用和案例
更多内容请见: 爬虫和逆向教程-专栏介绍和目录 文章目录 1. Feapder 概述1.1 Feapder介绍1.2 Feapder 核心特点1.3 Feapder 主要组件1.4 Feapder的安装2. 基础爬虫编写2.1 创建爬虫2.2 运行爬虫3. 数据采集案例3.1 新闻网站采集3.2 电商商品采集3.3 使用 Spider 类创建更强大爬…...
mybatis里in关键字拼接id问题
我们一般会把ids集合用StrUtil.join(‘,’)转成"1,2,3"这种形式 然后放入in中 我们会这么写: select id, nick_name, icon from tb_user where id in (#{ids}) order by FIELD(id, #{ids})结果发现sql执行是这样的: select id, nick_name, icon from tb_user where…...
在rockylinux9.4安装mongodb报错:缺少:libcrypto.so.10文件库
问题点: rockylinux9.4系统环境报错: ./mongod: error while loading shared libraries: libcrypto.so.10: cannot open shared object file: No such file or directory 解决方法: Ps:解压之后,检查mongodb的依赖环境…...
Spring Boot集成阿里云OSS:对象存储实战指南
Spring Boot集成阿里云OSS:对象存储实战指南 1. OSS是什么?为什么选择阿里云OSS? 对象存储(OSS) 是一种用于存储非结构化数据(如图片、视频、日志文件)的云服务,核心功能包括&#…...
【力扣hot100题】(019)旋转图像
比较考验脑子转不转得过来,最好先在纸上画一下图整理思路,不要和我一样上来就无脑套循环。 理解了思路还是好做的,每个小循环转一圈,大循环代表转的第几圈。小循环循环n-2i-1次,大循环循环(n1)…...
06_约束
文章目录 一、是什么二、实体完整性约束2.1、主键约束2.2、主键自增长2.3、唯一约束 三、域完整性约束3.1、非空约束3.2、默认值 四、引用完整性约束 一、是什么 用于限制加入表的数据的类型和规范,约束是添加在列上的,用来约束列的。 分类: …...
Anolis OS 8.4修复CVE-1999-0554漏洞记录
1. 使用TCP Wrappers配置白名单 通过修改/etc/hosts.allow和/etc/hosts.deny文件,仅允许特定IP访问NFS的mountd服务(需确保系统支持TCP Wrappers): 编辑/etc/hosts.allow,添加允许的客户端IP(如192.168.1…...
Seata AT模式的一些常见问题及其源码解析
Seata 是一款开源的分布式事务解决方案,致力于提供高性能和简单易用的分布式事务服务。Seata 将为用户提供了 AT、TCC、SAGA 和 XA 事务模式,为用户打造一站式的分布式解决方案。 Seata AT 基于两阶段提交协议的演变: 一阶段:业…...
华为GaussDB数据库的手动备份与还原操作介绍
数据库的备份以A机上的操作为例。 1、使用linux的root用户登录到GaussDB服务器。 2、用以下命令切换到 GaussDB 管理员用户,其中,omm 为当前数据库的linux账号。 su - omm 3、执行gs_dump命令进行数据库备份: 这里使用gs_dump命令进行备…...
2025年3月29日(matlab -ss -lti)
线性时不变系统(LTI系统)的定义与核心特性 线性时不变系统(Linear Time-Invariant System)是信号与系统分析中的基础模型,其核心特性包括线性和时不变性。以下从定义、验证方法和应用场景展开说明: 1. 线性…...
网络原理-TCP/IP
网络原理学习笔记:TCP/IP 核心概念 本文是我在学习网络原理时整理的笔记,主要涵盖传输层、网络层和数据链路层的核心协议和概念,特别是 TCP, UDP, IP, 和以太网。 一、传输层 (Transport Layer) 传输层负责提供端到端(进程到进…...
服务器磁盘卷组缓存cache设置介绍
工具1: storcli a. 确认软件包是否安装 [rootlocalhost ~]#rpm -qa | grep storcli storcli-1.21.06-1.noarch 备注:若检索结果为空,需要安装对应的软件安装包。安装命令如下: #rpm -ivh storcli-xx-xx-1.noarch.rpm b. 查看逻辑…...
Unity顶点优化:UV Splits与Smoothing Splits消除技巧
一、顶点分裂问题概述 1. 什么是顶点分裂 顶点分裂(Vertex Splits)是3D渲染中常见的性能问题,当模型需要为同一顶点位置存储不同属性值时,会创建多个顶点副本。主要分为两类: UV Splits:由UV不连续引起 Smoothing Splits&#…...
第五十三章 Spring之假如让你来写Boot——环境篇
Spring源码阅读目录 第一部分——IOC篇 第一章 Spring之最熟悉的陌生人——IOC 第二章 Spring之假如让你来写IOC容器——加载资源篇 第三章 Spring之假如让你来写IOC容器——解析配置文件篇 第四章 Spring之假如让你来写IOC容器——XML配置文件篇 第五章 Spring之假如让你来写…...
