DJ音乐,pcm鼓点计算

pcm怎么实时找出鼓点,歌曲边放边计算,现在能拿到pcm数据

https://www.cnblogs.com/8335IT/p/16190734.html

鼓点是指波形图吗,如果是的话可以参考链接里的算法

一般一个采样点,16位深,就是每个采样点2个字节;这个short>=0 就是正电平值;小于0就是负数电平值;

short s_value = * (buffer+ index);//buffer是一个short数组

计算的正向 DB值 = 20*log10((0.0+(int)s_value)/32767)

计算的负向 DB值 = 20*log10((0.0+(abs)s_value)/32767)

...

PCM声音是重采样为无符号16bit的深度的,然后我们需要得到某一时间(一般是零点几毫秒)PCM所在内存的地址和PCM声音的大小,而16bit也就是16bit/8bit=2byte,在c语言中2byte用short int来表示,因此我们可以从PCM所在地址里面按顺序取出2个byte的数据然后转化成short int的值就可以拿到当前采样点的振幅了
这样就可以获取到PCM声音的分贝值,绘制波形图了,通过波形图,计算PCM鼓点。
代码示例:

/**
 * 获取所有振幅之平均值 计算db (振幅最大值 2^16-1 = 65535 最大值是 96.32db)
 * 16 bit == 2字节 == short int
 * 无符号16bit:96.32=20*lg(65535);
 * 
 * @param pcmdata 转换成char类型,才可以按字节操作
 * @param size pcmdata的大小
 * @return
 */
int Audio::getPcmDB(const unsigned char *pcmdata, size_t size) {
 
    int db = 0;
    short int value = 0;
    double sum = 0;
    for(int i = 0; i < size; i += 2)
    {
        memcpy(&value, pcmdata+i, 2); //获取2个字节的大小(值)
        sum += abs(value); //绝对值求和
    }
    sum = sum / (size / 2); //求平均值(2个字节表示一个振幅,所以振幅个数为:size/2个)
    if(sum > 0)
    {
        db = (int)(20.0*log10(sum));
    }
    return db;
}

代码参考自链接:https://blog.csdn.net/tanningzhong/article/details/100985269

。。。。。。

供参考:
【音视频数据处理入门:PCM采样数据处理】https://mbd.baidu.com/ma/s/wVEmPivy