【Android 高性能音频】AAudio 音频流 PCM 采样 的 采样 缓冲 播放 的 连续机制 ( 数据回调机制 | 数据回调函数指针 | 实现数据回调函数 | 设置数据回调函数 )

简介: 【Android 高性能音频】AAudio 音频流 PCM 采样 的 采样 缓冲 播放 的 连续机制 ( 数据回调机制 | 数据回调函数指针 | 实现数据回调函数 | 设置数据回调函数 )

文章目录

I . AAudio 音频流 采样 缓冲 播放 的连续机制

II . AAudio 音频流 数据回调函数 函数指针类型定义

III . AAudio 音频流 数据回调函数 实现

IV . AAudio 音频流 数据回调函数 设置



I . AAudio 音频流 采样 缓冲 播放 的连续机制


1 . AAudio 音频流的 采样 缓冲 播放 流程 : 样本采样完成后 , 存入缓冲区 , 然后将其通过 AAudio 播放出来 , 采样阶段采集 n nn 个样本 , 然后将其放入缓冲区 , 将缓冲区的数据 写出到 AAudio 音频流中播放出来 ;



2 . 采样速度高于播放速度 : 如果采样采集多了 , 不能立刻播放 , 此时就会产生延迟 , 并且如果超出缓冲区大小 , 超出部分采样就会溢出 , 造成数据损失 , 样本不连续 , 就会产生电流 ;



3 . 采样速度低于播放速度 : 如果采样少了 , 不能向 AAudio 音频流中写入足够的数据 , 就会造成电流杂音等情况 ;



4 . 数据回调函数 引入 : 数据回调函数就是为了解决上述问题 , 引入的机制 ;



5 . 数据回调函数 简介 :


① 采样缓冲 : 采样后 , 将采集的样本存入缓冲区 ;

② 播放采样 : 将缓冲区中的样本写入 AAudio 音频流 ;

③ 调用回调函数 : AAudio 音频流如果播放完当前数据 , AAudio 就会自动调用 开发者按照 规范开发的 回调函数 申请后续采样数据 ;

④ 回调函数内容 : 开发者自己实现该回调函数 , 在这个函数中实现采样 并将采样设置给 AAudio 音频流 , 之后继续播放音频采样 ;

之后如果采样播放完毕 , 继续调用回调函数 ;



下面会着重讲解该数据回调函数的细节




II . AAudio 音频流 数据回调函数 函数指针类型定义


数据回调函数原型 : AAudio 只定义了一个函数类型 , 该函数的实际内容需要开发者自己开发 , 一般是 采样 , 然后 设置数据给 AAudio 音频流 的操作 ;


t

ypedef aaudio_data_callback_result_t (*AAudioStream_dataCallback)(
        AAudioStream *stream,
        void *userData,
        void *audioData,
        int32_t numFrames);


① 函数类型 : 该数据回调函数类型如下 , 其返回值是 aaudio_data_callback_result_t 类型 , 四个参数分别如下 ;

aaudio_data_callback_result_t (*)(
        AAudioStream *stream,
        void *userData,
        void *audioData,
        int32_t numFrames)


② 函数别名 : 使用 typedef 关键字为 上述函数指针类型 赋予一个类型别名 AAudioStream_dataCallback ;

③ 参数一 AAudioStream *stream : AAudio 音频流指针 ;

④ 参数二 void *userData : 该参数用于传递一些额外数据 , 与 AAudioStreamBuilder_setCallback() 中的 第三参数 void *userData 参数指向的地址一致 ;

⑤ 参数三 void *audioData : 指向音频采样数据的指针 , AAudio 会自动将该数据输出或输入到音频流中 ;

⑥ 参数四 int32_t numFrames : 要处理的帧数 , 需要将多少帧的 audioData 指针指向的音频采样数据 , 输入 或 输出到 AAudio 音频流中 ;



III . AAudio 音频流 数据回调函数 实现


aaudio_data_callback_result_t (*)(
        AAudioStream *stream,
        void *userData,
        void *audioData,
        int32_t numFrames)


1 . 数据回调函数设置给 AAudio 音频流 : AAudio 中通过 AAudioStreamBuilder_setDataCallback() 函数 , 将用户自己实现的 AAudioStream_dataCallback 回调函数的函数指针设置给 AAudio 音频流 , 当 AAudio 音频流需要数据时会自动回调该函数 ;



2 . 输出流回调函数实现内容 : 在该函数中需要 按照 AAudio 音频流的当前数据格式 ( 通道数/每帧样本数 , 采样率 ) , 采集 numFrames 帧的 PCM 音频样本数据 ( 每帧的采样数与通道数一致 ) , 将这些样本数据写出到 void *audioData 指针指向的内存中 , 之后这些数据会被自动输出到 AAudio 音频流中 ;



3 . 输入流回调函数实现内容 : 在函数中需要从 void *audioData 指针指向的内存中 , 读取 numFrames 帧 ( 每帧的采样数与通道数一致 ) 的采样数据 , 注意需要按照当前的 采样格式 ( 通道数/每帧样本数 , 采样率 ) 计算读取的字节大小 ;



4 . 采样数据自动传输 ( 不需要手动干预 ) : 在回调函数中 , 将 numFrames 帧的数据传递给 void *audioData , AAudio 在该回调函数执行完毕后 , 会自动将这些数据 读/写 到 AAudio 音频流中 , 不需要 开发者 手动调用 AAudioStream_read() 或 AAudioStream_write() 方法 , 一调用必出错 ;



5 . 每次读写的帧数 int32_t numFrames :


① 固定帧数 : 通过调用 AAudioStreamBuilder_setFramesPerDataCallback() 方法可以设置每次回调都读写固定帧数的音频采样 ;

② 自由帧数 : 如果用户没有指定帧数 , 那么在每次回调函数中的 numFrames 帧数可以由用户自己设置 ;


6 . 不能执行耗时操作 : 在该回调函数中 , 不能执行太耗时的操作 或 阻塞操作 , 如果阻塞时间超过了采样播放的时间 , 就会造成后续采样无法及时 读取 或 写入 到 AAudio 音频流中 , 出现音频故障 ;



7 . 回调函数中不能执行的操作 : 该回调函数的回调频率很高 , 可能达到每秒几百到几千次 , 因此有很多 耗时操作 或 访问本地资源 的逻辑不能再该函数中运行 , 尽可能只对内存数据进行操作 ;


① 内存操作 : 使用 malloc() 或 new 分配堆内存 , 极大可能造成内存泄漏或内存溢出 ;

② 文件操作 : 打开 open , 关闭 close , 读取 read , 写出 write 等针对文件的操作 ;

③ 网络操作 : 访问网络操作 , 从网络中读取数据 , 或向远程端口发送数据 ;

④ 同步线程 : 线程间的同步操作会造成阻塞 ;

⑤ 休眠阻塞 : sleep 方法不能执行 , 会造成阻塞 ;

⑥ 关音频流 : 停止 或 关闭 流操作 会造成不可预知故障 ;

⑦ 读写操作 : 该函数中不用刻意调用 AAudioStream_read() 和 AAudioStream_write() 方法进行读写操作 ;


8 . 回调函数中可以进行的操作 :


① 调用 AAudioStream_getXXX() 类方法 : 如下图中列举的方法可以直接调用 , 获取 AAudio 音频流的各种属性 ;

image.png

② 调用 AAudio_convertResultToText() 方法 : 将返回值转为对应的 ASCII 字符 ;


9 . 非阻塞技术 : 如果需要在回调函数中 读取 或 输出 数据 , 建议使用非阻塞技术 , 如 FIFO 技术 ;




IV . AAudio 音频流 数据回调函数 设置


1 . 数据回调函数设置方法 :


① 函数原型 : 该方法用于设置 AAudio 音频流回调函数 , 当 AAudio 需要 读取 / 写出数据时 , 会自动回调该 AAudioStream_dataCallback 类型 函数 ;

AAUDIO_API void AAudioStreamBuilder_setDataCallback(
  AAudioStreamBuilder *builder,
  AAudioStream_dataCallback callback,
  void *userData


② 参数 一 AAudioStreamBuilder *builder : AAudio 音频流指针 ;

③ 参数 二 AAudioStream_dataCallback callback : AAudioStream_dataCallback 是函数指针类型 , 开发者自己实现该函数 , 然后将函数的地址当做参数设置到此处 ;

typedef aaudio_data_callback_result_t (*AAudioStream_dataCallback)(
        AAudioStream *stream,
        void *userData,
        void *audioData,
        int32_t numFrames);


④ 参数 三 void *userData : 该指针 与 AAudioStream_dataCallback 函数指针类型的第二个参数指向的地址是相同的 , 相当于传入一个用户自定义的指针 , 可以是任意类型任意数据任意变量 ;

2 . 音频数据传递方式 : 在 AAudioStream_dataCallback 函数指针类型的回调函数中 , 音频流的数据不是通过 AAudioStream_read 或 AAudioStream_write 方法进行读写的 , 只要将音频采样数据设置给函数的第三个参数 audioData 指针即可 , 第四个参数 numFrames 用于说明读写数据的帧数 , 每帧的采样数 就是 采样的通道数 ;


3 . 数据回调函数工作机制 :


① 第一次回调 : 在 AAudio 音频流调用 AAudioStream_requestStart() 方法后 , 会立刻回调该数据回调函数 , 然后第一次 读写采样数据到 AAudio 音频流中 ;

② 循环回调 : 当 AAudio 音频流 读取或写出数据完毕后 , 会自动回调该数据回调函数 , 在回调函数中准备下一次的采样 , 读写到 AAudio 音频流中 , 之后继续循环 , 直到 AAudio 音频流关闭销毁 ;

③ 实时线程 : AAudio 拥有一个实时线程 , 该数据回调函数就是运行在这个线程上的 ;


目录
相关文章
|
Java 程序员 开发工具
Android|修复阿里云播放器下载不回调的问题
虽然 GC 带来了很多便利,但在实际编码时,我们也需要注意对象的生命周期管理,该存活的存活,该释放的释放,避免因为 GC 导致的问题。
264 2
|
编解码 网络协议 开发工具
Android平台如何实现多路低延迟RTSP|RTMP播放?
本文档详细介绍了大牛直播SDK在Android平台上实现RTSP与RTMP流媒体播放及录像功能的技术细节。早在2015年,SDK的第一版就已经支持了多实例播放,并且通过简单的实例封装就能轻松实现。文档中提供了代码示例,展示了如何开启播放、停止播放以及开始和停止录像等功能。此外,SDK还提供了丰富的配置选项,例如设置录像目录、文件大小限制、转码选项等。总结部分列出了该SDK的关键特性,包括但不限于高稳定性和低延迟的播放能力、多实例支持、事件回调、硬解码支持、网络状态监控以及复杂的网络环境处理等。这些功能使得SDK能够应对各种应用场景,特别是在对延迟和稳定性有极高要求的情况下表现优异。
582 5
|
Android开发
Android 利用MediaPlayer实现音乐播放
本文提供了一个简单的Android MediaPlayer音乐播放示例,包括创建PlayerActivity、配置AndroidManifest.xml和activity_player.xml布局,以及实现播放和暂停功能的代码。
463 0
Android 利用MediaPlayer实现音乐播放
|
编解码 网络协议 vr&ar
Android平台下VR头显如何低延迟播放4K以上超高分辨率RTSP|RTMP流
这段内容讲述了VR头显中实现高分辨率视频播放的技术背景与实现方法,并强调了其重要性。高分辨率对于提升VR体验至关重要,它能提供更清晰的画面、增强沉浸感、补偿透镜放大效应,并维持宽广视场角下的图像质量。文中提到的大牛直播SDK具备极低的延迟(200-400ms),支持多种协议与格式,并具有丰富的功能特性,如多实例播放、事件回调、视频及音频格式支持等。此外,提供了基于Unity的播放器示例代码,展示了如何配置播放参数并开始播放。最后,作者指出此类技术在远程控制、虚拟仿真等应用场景中的重要意义。
380 3
|
算法 数据处理 开发工具
Android平台RTSP|RTMP播放器如何回调YUV或RGB数据
在开发Android平台上的RTSP或RTMP播放器时,开发者不仅追求低延迟播放,还希望获取解码后的视频数据(如YUV或RGB格式),以便进行视觉算法分析。使用大牛直播SDK中的SmartPlayer,可在确保播放流畅的同时,通过设置外部渲染器(`SmartPlayerSetExternalRender`)来高效地回调原始视频数据。例如,对于RGBA数据,需实现`NTExternalRender`接口,并重写相关方法以处理数据和尺寸变化。同样地,对于I420(YUV)数据,也需要相应地实现接口以满足需求。这种方式使得开发者能在不影响常规播放功能的情况下,进行定制化的视频处理任务。
326 1
|
开发工具 Android开发
Android项目架构设计问题之SDK内部减少每次回调时的冗余判断逻辑如何解决
Android项目架构设计问题之SDK内部减少每次回调时的冗余判断逻辑如何解决
221 0
|
开发工具 Android开发
Android项目架构设计问题之外部客户方便地设置回调如何解决
Android项目架构设计问题之外部客户方便地设置回调如何解决
206 0
|
Java API 开发工具
Android项目架构设计问题之为SDK添加新的回调支持如何解决
Android项目架构设计问题之为SDK添加新的回调支持如何解决
232 0
|
数据采集 Android开发 索引
【Android RTMP】音频数据采集编码 ( AAC 音频格式解析 | FLV 音频数据标签解析 | AAC 音频数据标签头 | 音频解码配置信息 )(二)
【Android RTMP】音频数据采集编码 ( AAC 音频格式解析 | FLV 音频数据标签解析 | AAC 音频数据标签头 | 音频解码配置信息 )(二)
801 0
【Android RTMP】音频数据采集编码 ( AAC 音频格式解析 | FLV 音频数据标签解析 | AAC 音频数据标签头 | 音频解码配置信息 )(二)
|
数据采集 存储 传感器
【Android RTMP】音频数据采集编码 ( AAC 音频格式解析 | FLV 音频数据标签解析 | AAC 音频数据标签头 | 音频解码配置信息 )(一)
【Android RTMP】音频数据采集编码 ( AAC 音频格式解析 | FLV 音频数据标签解析 | AAC 音频数据标签头 | 音频解码配置信息 )(一)
496 0
【Android RTMP】音频数据采集编码 ( AAC 音频格式解析 | FLV 音频数据标签解析 | AAC 音频数据标签头 | 音频解码配置信息 )(一)