【Android 高性能音频】AAudio 音频流 样本缓冲 相关配置 ( 通道数 | 样本格式 | 帧缓冲 | 采样率 | 每帧样本数 == 通道数 )

2023-03-27 17:34:43 浏览数 (1)

文章目录
  • I . AAudio 音频流创建流程
  • II . AAudio 音频流构建器 设置 通道数 AAudioStreamBuilder_setChannelCount
  • III . AAudio 音频流构建器 设置音频格式 AAudioStreamBuilder_setFormat
  • IV . AAudio 音频流 样本格式
  • V . AAudio 音频流 设置缓冲区大小 AAudioStream_setBufferSizeInFrames
  • VI . AAudio 音频流 采样率设置 AAudioStreamBuilder_setSampleRate
  • VII . AAudio 音频流 获取当前采样率 AAudioStream_getSampleRate
  • VIII . AAudio 音频流 每帧采样数

I . AAudio 音频流创建流程

红色标题是本博客讲解的内容 , 黑色是前几篇讲过的内容 ;

使用 AAudio 音频库 , 首先需要导入 AAudio.h 头文件 ;

代码语言:javascript复制
#include <AAudio.h>

创建 AAudio 音频流 , 需要先创建 AAudio 音频流构建器 , 然后在通过该构建器创建音频流 ;

代码语言:javascript复制
    //创建构建器 , AAudio 音频流通过该构建器创建
    //声明 AAudio 音频流构建器 指针
    AAudioStreamBuilder *builder = nullptr;
    //创建 AAudio 音频流构建器 , 注意传入二维指针
    aaudio_result_t result = AAudio_createStreamBuilder(&builder);

设置音频设备 ID ;

代码语言:javascript复制
    // 设置音频流设备 ID
    AAudioStreamBuilder_setDeviceId(builder, playbackDeviceId_);

设置音频流方向 ;

代码语言:javascript复制
    // 设置音频流方向
    AAudioStreamBuilder_setDirection(builder, AAUDIO_DIRECTION_OUTPUT);

设置音频设备共享模式 ;

代码语言:javascript复制
    // 设置共享模式 , 独占模式性能更高 , 延迟更低 ; 如果 该音频设备正在被使用 , 设置失败会自动设置成 共享模式
    AAudioStreamBuilder_setSharingMode(builder, AAUDIO_SHARING_MODE_EXCLUSIVE);

设置性能模式 ;

代码语言:javascript复制
    // 设置性能模式
    AAudioStreamBuilder_setPerformanceMode(builder, AAUDIO_PERFORMANCE_MODE_LOW_LATENCY);

设置 AAudio 音频流通道数 :

代码语言:javascript复制
    // 设置通道个数
    AAudioStreamBuilder_setChannelCount(builder, sampleChannels_);

设置 AAudio 音频流样本格式 :

代码语言:javascript复制
    // 设置音频格式
    AAudioStreamBuilder_setFormat(builder, sampleFormat_);

设置 AAudio 音频流缓冲区大小 : 这里的缓冲区是播放器的缓冲区 , 单位是帧 , 每帧的采样数就是通道数 , 单声道 每帧 1 个采样, 双声道立体声每帧 2 个采样 , 分别对应左右声道的采样 ;

代码语言:javascript复制
    // 设置每帧的缓冲区大小 , 可以通过该设置达到尽可能低的延迟
    AAudioStream_setBufferSizeInFrames(playStream_, framesPerBurst_);

下面会着重对上面的流程细节进行详细解析 ; 每个方法的参数 , 原理 , 返回值 等细节都会讲解到 ;

II . AAudio 音频流构建器 设置 通道数 AAudioStreamBuilder_setChannelCount

1 . AAudio 音频流通道数设置 :

  • ① 函数原型 :
代码语言:javascript复制
AAUDIO_API void AAudioStreamBuilder_setChannelCount(
  AAudioStreamBuilder *builder,
  int32_t channelCount
)
  • ② 函数作用 : 设置音频流的通道数 , 设置 1 代表 单声道 , 设置 2 代表 立体声 ;
  • ③ 代码示例 :
代码语言:javascript复制
    // 设置通道个数
    AAudioStreamBuilder_setChannelCount(builder, sampleChannels_);

2 . 默认处理 :

  • ① 默认值 : 如果没有调用该函数设置通道数 , 默认为 AAUDIO_UNSPECIFIED ;
  • ② 默认情况处理 : 如果通道数未指定 , 打开流时系统自动选择一个最佳通道数 , 不同设备可能有不同的通道数 ;

3. 指定通道值情况处理 : 如果指定了通道数 , 那么打开流时会使用该通道数 ; 如果通道数与设备不匹配 , 那么 AAudio 音频流打开时会报错 ;

III . AAudio 音频流构建器 设置音频格式 AAudioStreamBuilder_setFormat

1 . AAudio 音频流格式设置 :

  • ① 函数原型 :
代码语言:javascript复制
AAUDIO_API void AAudioStreamBuilder_setFormat(
  AAudioStreamBuilder *builder,
  aaudio_format_t format
)
  • ② 函数作用 : 设置 AAudio 音频流的样本数据格式 ;
  • ③ 参数 AAudioStreamBuilder *builder : AAudio 音频流构建器 ;
  • ④ 参数 aaudio_format_t format : 音频格式 , 一般是 AAUDIO_FORMAT_PCM_FLOAT ( 浮点型采样格式 ) 或 AAUDIO_FORMAT_PCM_I16 ( 每个采样 16 位 ) ;
  • ⑤ 代码示例 :
代码语言:javascript复制
    // 设置音频格式
    AAudioStreamBuilder_setFormat(builder, sampleFormat_);

2 . 默认设置 :

  • ① 默认值 : 如果没有设置音频流样本格式 , 那么会默认设置成 AAUDIO_UNSPECIFIED ;
  • ② 默认值打开流 : 默认值 AAUDIO_UNSPECIFIED 状态下 , 如果打开 AAudio 音频流 , 系统会自动选择一个最佳的样本格式 , 这个值可能由 Android 设备决定 , 每个设备的值可能都不一样 ;
IV . AAudio 音频流 样本格式

AAudio 音频流 样本格式 :

  • ① 样本格式定义 : aaudio_format_t 是 int32_t 类型数据 , 该枚举有四种取值 ;
代码语言:javascript复制
enum {
    AAUDIO_FORMAT_INVALID = -1,
    AAUDIO_FORMAT_UNSPECIFIED = 0,

    /**
     * This format uses the int16_t data type.
     * The maximum range of the data is -32768 to 32767.
     */
    AAUDIO_FORMAT_PCM_I16,

    /**
     * This format uses the float data type.
     * The nominal range of the data is [-1.0f, 1.0f).
     * Values outside that range may be clipped.
     *
     * See also 'floatData' at
     * https://developer.android.com/reference/android/media/AudioTrack#write(float[], int, int, int)
     */
    AAUDIO_FORMAT_PCM_FLOAT
};
typedef int32_t aaudio_format_t;
  • ② AAUDIO_FORMAT_UNSPECIFIED 格式 : 未定义格式 , 默认的格式 , 如果音频流打开系统跟自动选择合适的采样格式 ;
  • ③ AAUDIO_FORMAT_PCM_I16 格式 : 每个样本是 16 位 , 其取值范围是 -32768 ~ 32767 ;
  • ④ AAUDIO_FORMAT_PCM_FLOAT 格式 : 样本由浮点型表示 , 取值范围 -1.0 ~ 1.0 ;
V . AAudio 音频流 设置缓冲区大小 AAudioStream_setBufferSizeInFrames

1. 函数作用 : 在音频流播放时 , 有可能会产生阻塞 , 即 采样播放完毕 , 新采样还没到达 , 该函数可以 通过 改变 缓冲区大小阈值 , 调整 缓冲区的延迟 , 即 如果出现 阻塞 , 可以增大该缓冲区大小 ( 帧数 ) ;

2. 结合 XRun 值使用 : 通过 AAudioStream_getXRunCount() 方法 , 可以获取 欠载 ( UnderRun ) 或 超限 ( OverRun ) 的值 , 根据该 XRun 值进行缓冲区大小的调整 , 达到为每个音频设备设置合适的延迟的目的 ;

3. 可设置的最大值 : 通过 AAudioStream_getBufferCapacityInFrames() 函数可以获取 缓冲区可设置的最大帧数 , 设置帧数时 , 不能超过该数值 ;

4. 查看当前缓冲区大小 : 调用 AAudioStream_getBufferSizeInFrames() 方法 , 可以查看当前的缓冲区帧数 ;

文档中的说法是 : 获取 AAudio 音频流在不阻塞的情况下 , 可以读取 或 写入的最大帧数 , 理解不通 ;

5. AAudioStream_setBufferSizeInFrames 函数简介 :

  • ① 函数原型 : numFrames 是设置的新的缓冲区帧数 , stream 代表 AAudio 音频流指针 ;
代码语言:javascript复制
AAUDIO_API aaudio_result_t AAudioStream_setBufferSizeInFrames(
  AAudioStream *stream,
  int32_t numFrames
)
  • ② 代码示例 :
代码语言:javascript复制
        //设置当前缓冲区是多少帧
        bufferSize = AAudioStream_setBufferSizeInFrames(stream, bufferSize);

播放器缓冲区 : 这里的帧缓冲区指的是 AAudio 音频流的缓冲区 , 属于播放器 或 音频设备 的固有属性 ; 采样缓冲区 : 注意与采样缓冲区进行区分 , 采样缓冲区指的是 一次性向 AAudio 音频流 读取 或 写入的 字节数 , 注意区分这两个缓冲区 ; 电流产生 : 如果两个缓冲区设置不当 , 会造成音频卡顿 , 电流 , 刺啦 或者 啪啪 的声音 ; 帧大小 : 这里的帧可以理解成一个样本 , 如果是单声道 , 每帧一个样本 , 如果是双声道立体声 , 每帧 2 个样本 ;

VI . AAudio 音频流 采样率设置 AAudioStreamBuilder_setSampleRate

推荐使用默认设置 , 不要调用该方法 设置采样率 , 获取默认的最佳采样率 , 然后根据该采样率进行采样即可 ;

1 . AAudioStreamBuilder_setSampleRate 方法简介 :

  • ① 函数原型 : 设置 AAudio 音频流的采样率 ;
代码语言:javascript复制
AAUDIO_API void AAudioStreamBuilder_setSampleRate(
  AAudioStreamBuilder *builder,
  int32_t sampleRate
)
  • ② sampleRate 参数 : sampleRate 参数就是采样率 , 其单位 赫兹 ( Hz ) , 一般情况下是 44100 Hz 或 48000 Hz , 代表一秒钟有 44100 或 48000 个采样 ;

2 . 最佳实践 :

  • ① 默认值 : 默认情况下 , 如果不调用该函数设置 AAudio 播放器的采样率 , 其值为 AAUDIO_UNSPECIFIED ;
  • ② 默认行为 : 在默认状态下 , 如果打开 AAudio 音频流 , 系统会自动分配一个最佳值 ;
  • ③ 应用开发推荐 : 打开默认采样率的音频流后 , 调用 AAudioStream_getSampleRate() 可以获取当前 AAudio 音频流的采样率 , 我们可以根据该采样率进行采样 ;

这也是我们推荐的做法 , 直接使用默认值即可 , 如果设置的值不合适 , 会造成 AAudio 音频流打开失败的后果 ;

  • ④ 出错情况 : 如果指定的采样率与音频设备不一致 , 打开音频流会失败 ;
VII . AAudio 音频流 获取当前采样率 AAudioStream_getSampleRate

AAudioStream_getSampleRate 简介 :

  • ① 函数原型 : 该函数用于获取当前 AAudio 音频流 采样率 ;
代码语言:javascript复制
AAUDIO_API int32_t AAudioStream_getSampleRate(
  AAudioStream *stream
)
  • ② 代码示例 :
代码语言:javascript复制
            // 获取音频流采样率
            sampleRate_ = AAudioStream_getSampleRate(playStream_);
VIII . AAudio 音频流 每帧采样数

设置每帧采样数 AAudioStreamBuilder_setSamplesPerFrame() :

  • ① 等同方法 : 该方法 与 AAudioStreamBuilder_setChannelCount() 方法 作用是相同的 ;
  • ② 每帧采样数 : 该值就是通道数 , 如果是单声道 , 每帧只有一个采样 , 如果是 双声道立体声 , 每帧有 2 个采样 ;
  • ③ 函数原型 :
代码语言:javascript复制
AAUDIO_API void AAudioStreamBuilder_setSamplesPerFrame(
  AAudioStreamBuilder *builder,
  int32_t samplesPerFrame
)

获取每帧采样数 AAudioStream_getSamplesPerFrame :

  • ① 等同方法 : 该方法 与 AAudioStream_getChannelCount() 方法 作用是相同的 ;
  • ② 每帧采样数 : 该值就是通道数 , 如果是单声道 , 每帧只有一个采样 , 如果是 双声道立体声 , 每帧有 2 个采样 ;
  • ③ 函数原型 :
代码语言:javascript复制
AAUDIO_API int32_t AAudioStream_getSamplesPerFrame(
  AAudioStream *stream
)

0 人点赞