原始音频数据
本文介绍如何对采集到的原始音频数据进行前处理和后处理。
技术原理
音频传输过程中,我们可以对采集到的音频数据进行前处理和后处理,获取想要的播放效果。
对于有自行处理音频数据需求的场景,声网提供原始数据功能。你可以在将数据发送给编码器前进行前处理,对捕捉到的音频信号进行修改;也可以在将数据发送给解码器后进行后处理,对接收到的音频信号进行修改。
API 调用时序
下图展示使用原始音频数据的调用时序:
前提条件
在进行操作之前,请确保你已经在项目中实现了基本的实时音视频功能。详见实现音视频互动。
实现方法
参考如下步骤,在你的项目中实现原始音频数据功能:
- 加入频道前,创建一个
IAudioFrameObserver
实例,并调用registerAudioFrameObserver
方法注册音频观测器。 - 调用
set
为前缀的方法配置音频帧格式。 - 实现
onRecordAudioFrame
、onPlaybackAudioFrame
、onPlaybackAudioFrameBeforeMixing
和onMixedAudioFrame
回调。 以上回调会对音频帧进行采集和处理。 如果以上回调的返回值是false
,则说明对音频帧的处理无效。
C++
BOOL CAgoraOriginalAudioDlg::RegisterAudioFrameObserver(BOOL bEnable, IAudioFrameObserver *audioFrameObserver)
{
agora::util::AutoPtr<agora::media::IMediaEngine> mediaEngine;
// 查询 AGORA_IID_MEDIA_ENGINE 接口
mediaEngine.queryInterface(m_rtcEngine, agora::rtc::AGORA_IID_MEDIA_ENGINE);
int nRet = 0;
if (mediaEngine.get() == NULL)
return FALSE;
if (bEnable)
// 注册音频观测器并传入 IAudioFrameObserver 对象
nRet = mediaEngine->registerAudioFrameObserver(audioFrameObserver);
else
// 取消注册音频观测器
nRet = mediaEngine->registerAudioFrameObserver(NULL);
return nRet == 0 ? TRUE : FALSE;
}
// 实现 onRecordAudioFrame 回调
bool COriginalAudioProcFrameObserver::onRecordAudioFrame(const char* channelId, AudioFrame& audioFrame)
{
SIZE_T nSize = audioFrame.channels * audioFrame.samplesPerChannel * 2;
unsigned int readByte = 0;
int timestamp = GetTickCount();
short *pBuffer = (short *)audioFrame.buffer;
for (SIZE_T i = 0; i < nSize / 2; i++)
{
if (pBuffer[i] * 2 > 32767) {
pBuffer[i] = 32767;
}
else if (pBuffer[i] * 2 < -32768) {
pBuffer[i] = -32768;
}
else {
pBuffer[i] *= 2;
}
}
#ifdef _DEBUG
CString strInfo;
strInfo.Format(_T("audio Frame buffer size:%d, timestamp:%d \n"), nSize, timestamp);
OutputDebugString(strInfo);
audioFrame.renderTimeMs = timestamp;
#endif
return true;
}
// 实现 onPlaybackAudioFrame 回调
bool COriginalAudioProcFrameObserver::onPlaybackAudioFrame(const char* channelId, AudioFrame& audioFrame)
{
return true;
}
// 实现 onMixedAudioFrame 回调
bool COriginalAudioProcFrameObserver::onMixedAudioFrame(const char* channelId, AudioFrame& audioFrame)
{
return true;
}
// 实现 onPlaybackAudioFrameBeforeMixing 回调
bool COriginalAudioProcFrameObserver::onPlaybackAudioFrameBeforeMixing(const char* channelId, rtc::uid_t uid, AudioFrame& audioFrame)
{
return true;
}
// 调用 set 为前缀的方法,配置每个回调采集到的音频帧
m_rtcEngine->setRecordingAudioFrameParameters(44100, 2, RAW_AUDIO_FRAME_OP_MODE_READ_WRITE, 1024);
m_rtcEngine->setPlaybackAudioFrameParameters(44100, 2, RAW_AUDIO_FRAME_OP_MODE_READ_WRITE, 1024);
m_rtcEngine->setPlaybackAudioFrameBeforeMixingParameters(44100, 2);
m_rtcEngine->setMixedAudioFrameParameters(44100, 2, 1024);
参考信息
本节介绍本文中使用方法的更多信息以及相关页面的链接。
示例项目
声网在 GitHub 上提供一个开源的示例项目:OriginalAudio