(三)音 频 数 据 的 压 缩
下 面 说 明 使 用 CODEC 实 现 音 频 压 缩 的 过 程;假 设 源 信 号 为8K 采 样、16bits PCM 编 码、 单 声 道、 长 度 为1 秒 的 音 频 信 号。 驱 动 程 序 采 用Windows 95 自 带 的TrueSpeech 音 频CODEC, 它 能 实 现 大 约10:1 的 压 缩。 在 此 例 中,TrueSpeech CODEC 支 持 从 源 音 频 格 式 到 目 标 格 式 的 转 换, 而 在 实 际 应 用 中, 可 能 某 种CODEC 不 支 持 直 接 将 源 音 频 格 式 转 换 成 目 标 格 式, 这 时 可 以 采 取 两 步 转 换 法, 即 先 将 源 格 式 转 换 成 一 种 中 间 格 式, 再 将 此 中 间 格 式 转 换 成 目 标 格 式, 因 为 线 性PCM 编 码 最 为 简 单, 且 为 绝 大 多 数CODEC 所 支 持, 所 以 一 般 中 间 格 式 都 选 为 线 性PCM 格 式 的 一 种。
1.在 进 行 压 缩 之 前 首 先 需 要 确 定TrueSpeech 驱 动 程 序 的ID 值: 为 此 需 要 用 到acmDriverEnum() 函 数, 对 枚 举 到 的 每 一 个 驱 动 程 序, 由acmDriverEnum() 指 定 的 回 调 函 数 将 检 查 其 支 持 的 所 有 音 频 格 式, 若 其 中 包 括wFormatTag 值 为WAVE_FORMAT_DSPGROUP_TRUESPEECH 的 音 频 格 式, 则 此 驱 动 程 序 就 是 要 寻 找 的TrueSpeech CODEC, 它 所 支 持 的 第 一 种WAVE_FORMAT_DSPGROUP_TRUESPEECH 音 频 格 式 即 为 目 标 音 频 压 缩 格 式。
2.打 开 驱 动 程 序:
根 据 查 询 的 结 果, 设 hadID 为 TrueSpeech CODEC 的 ID 值,pwfDrv 为 指 向 目 标 WAVEFORMATEX 结 构 的 指 针, 接 下 来 利 用 获 得 的ID 值 打 开 相 应 的 驱 动 程 序
HACMDRIVER had = NULL;
mmr = acmDriverOpen(&had, hadID, 0);
if(mmr) { printf(" 打开驱动程序失败
"); exit(1); }
3.压 缩: 和 解 压 缩 一 样, 都 是 将 音 频 信 号 从 一 种 音 频 格 式 转 换 成 另 一 种 格 式, 要 完 成 这 一 过 程, 首 先 要 打 开 转 换 流; 在 用acmStreamOpen 打 开 转 换 流 时, 我 们 指 定 了ACM_STREAMOPENF_NONREALTIME 标 志, 它 表 示 转 换 无 需 实 时 进 行, 因 为 很 多 压 缩 算 法 的 计 算 量 是 相 当 大 的, 实 时 完 成 几 乎 是 不 可 能 的, 例 如 在 本 例 中, 如 果 不 指 定 此 标 志,TrueSpeech CODEC 就 会 返 回“ 无 法 完 成” 的 错 误。
HACMSTREAM hstr = NULL;
DWORD dwSrcBytes = dwSrcSamples * wfSrc.wBitsPerSample / 8;
mmr = acmStreamOpen(&hstr,had, //驱动程序句柄 pwfSrc, //指向源音频格式的指针 pwfDrv, //指向目标音频格式的指针 NULL, //无过滤器 NULL, //无回调函数 0,ACM_STREAMOPENF_NONREALTIME);
在 真 正 进 行 转 换 之 前, 还 必 须 准 备 转 换 流 的 信 息 头。 下 面 一 段 代 码 中, 先 利 用 源 数 据 的 大 小 以 及 目 标 格 式 的 平 均 数 据 率 估 算 目 标 数 据 的 缓 存 区 大 小, 然 后 调 用acmStreamPrepareHeader 为 转 换 准 备 信 息 头。
DWORD dwDstBytes=pwfDrv->nAvgBytesPerSec*dwSrcSamples/wfSrc.nSamplesPerSec;
dwDstBytes = dwDstBytes*3/2; // 计 算 压 缩 后 音 频 数 据 大 小, 并 依 此 适 当 增 加 输 出 缓 冲 区 的 大 小。
BYTE* pDstData = new BYTE [dwDstBytes];
ACMSTREAMHEADER shdr;
memset(&strhdr, 0, sizeof(shdr));
shdr.cbStruct = sizeof(shdr);
shdr.pbSrc = pSrcData; //源音频数据区
shdr.cbSrcLength = dwSrcBytes;
shdr.pbDst = pDstData; //压缩后音频数据缓冲区
shdr.cbDstLength = dwDstBytes;
mmr = acmStreamPrepareHeader(hstr, &shdr, 0);
语 音 数 据 真 正 的 压 缩 过 程 是 由 函 数acmStreamConvert() 完 成 的。 在 调 用acmStreamConvert() 时 可 以 指 定 回 调 函 数, 以 便 在 转 换 过 程 中 显 示 进 度 信 息 等。 在 本 例 中, 未 指 定 回 调 函 数, 只 是 简 单 地 等 待 压 缩 的 结 束。
—- mmr = acmStreamConvert(hstr, &shdr, 0);
数 据 压 缩 完 毕 后, 应 用 程 序 就 可 以 把 缓 冲 区 中 的 数 据 写 入 目 标 文 件 中。
最 后, 必 须 关 闭 转 换 流 和 驱 动 程 序。
mmr = acmStreamClose(hstr, 0); mmr = acmDriverClose(had, 0);