博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
PCM双声道分离为单声道
阅读量:6069 次
发布时间:2019-06-20

本文共 2189 字,大约阅读时间需要 7 分钟。

hot3.png

这里主要是写 的是 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播放;

 

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

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

 

转载于:https://my.oschina.net/sfshine/blog/1813480

你可能感兴趣的文章
poj 3984迷宫问题【广搜】
查看>>
oracle ORA-01840:输入值对于日期格式不够长
查看>>
python基础知识~logger模块
查看>>
SIP入门(二):建立SIPserver
查看>>
Servlet3.0的异步
查看>>
WebService连接postgresql( 失败尝试)
查看>>
从头认识java-13.11 对照数组与泛型容器,观察类型擦除给泛型容器带来什么问题?...
查看>>
Python-MacOSX下SIP引起的pip权限问题解决方案(非取消SIP机制)
查看>>
从MFQ方法到需求分析
查看>>
android.view.WindowManager$BadTokenException: Unable to add window
查看>>
HDU5012:Dice(bfs模板)
查看>>
iphone openssh
查看>>
Linux下MEncoder的编译
查看>>
Xamarin使用ListView开启分组视图Cell数据展示bug处理
查看>>
Javascript中闭包(Closure)的探索(一)-基本概念
查看>>
spark高级排序彻底解秘
查看>>
ylbtech-LanguageSamples-PartialTypes(部分类型)
查看>>
福建省促进大数据发展:变分散式管理为统筹集中式管理
查看>>
开发环境、生产环境、测试环境的基本理解和区别
查看>>
tomcat多应用之间如何共享jar
查看>>