PCM双声道分离为单声道

这里主要是写 的是 16位 双声道 PCM数据的分离;

可以看这篇文章查看双声道PCM的内存结构:

http://blog.csdn.net/ownwell/article/details/8114121

可以看出:

16位sample的PCM数据,每两个字节,一个字节包含一个升到的数据;

所以只需将隔一个字节获取一个字节的数据,就可以单独获取一个声道的数据了;如:获取 0,2,4,6,8.....字节的数据 和 1,3,5,7,9......字节的数据;

注意:

提取位单声道后:其它PCM音频参数不变,只是声道数由2变为1了;sample依然是16位的;可以通过audacity播放;

扫描二维码关注公众号,回复: 868869 查看本文章

C++的实现;

为了简单,我这里先知实现一个声道的获取:

unsigned char * get_oneChannel_left_from_doubleChannel(unsigned char * pDoubleChannelBuf, int nLen, int nPerSampleBytesPerChannle)
{
int nOneChannelLen = nLen / 2;


unsigned char * pOneChannelBuf = new unsigned char[nOneChannelLen];


for ( int i = 0 ; i < nOneChannelLen/2; i++ )
{
memcpy((uint16_t*)pOneChannelBuf + i, ((uint32_t *)(pDoubleChannelBuf)) + i, nPerSampleBytesPerChannle);
}


return pOneChannelBuf;
}



void CMFCDlg::OnBnClickedButton2()
{
// TODO: Add your control notification handler code here


char * pstrFile = "E:\\webrtc_Audio_Remote_BytesPerSample_2,__channels_2,__SampleRate_48000.pcm";


int nFilLen = file_size(pstrFile);
unsigned char * pFileBuf = new unsigned char[nFilLen];


FILE * f = fopen(pstrFile, "r+b");
int nread = fread(pFileBuf, 1, nFilLen, f);
fclose(f);


unsigned char * pOneChannelBuf = get_oneChannel_left_from_doubleChannel(pFileBuf, nFilLen, 2 ); 

delete[] pFileBuf;
delete[] pOneChannelBuf;

}

这里转一个网上的一个函数,也可以实现,但是代码不够简洁:

from:https://zhidao.baidu.com/question/1608973248205344987.html

BOOL CopyChannelData(
unsigned char * lpData,       // 需要处理的PCM数据
int nSize,                    // PCM数据字节长度
int nSampleSize,             // 音频采样大小,一般是8位,具体看自身文件
unsigned char * lpLeftData,  // 保存左声道数据缓冲,该缓冲不能小于nSize的一半,最好是和nSize相等
unsigned char * lpRightData  // 同上,保存右声道数据
)
{


char *pPtr1 = (char*)lpData;
char *pPtr2 = NULL;


int nCountLeft = 0, nCountRight = 0;


for (int i = 0; i < nSize; i += 2 * nSampleSize)
{
//左声道处理
memcpy((char*)lpLeftData + nCountLeft, pPtr1, nSampleSize);


nCountLeft += nSampleSize;
pPtr2 = pPtr1 + nSampleSize;

if ((DWORD)pPtr2 - (DWORD)lpData >= (DWORD)nSize)
{
break;
}


//右声道处理
memcpy((char*)lpRightData + nCountRight, pPtr2, nSampleSize);
nCountRight += nSampleSize;
pPtr1 = pPtr2 + nSampleSize;
}
  
return TRUE;
}

在webrtc中,可以用:webrtc58\src\webrtc\audio\utility\audio_frame_operations.h

具体可以参考相关相关目录的测试用法;

猜你喜欢

转载自my.oschina.net/sfshine/blog/1813480
今日推荐