12

我正在使用 EZAudio 构建一个 iOS 应用程序。它的委托返回一个float**缓冲区,其中包含指示检测到的音量的浮点值。这个委托被不断地调用,它的工作在不同的线程中完成。

我要做的是从 EZAudio 获取浮点值并将其转换为分贝。


EZAudioDelegate

这是我用于获取麦克风数据的简化 EZAudio 代表

- (void)microphone:(EZMicrophone *)microphone hasAudioReceived:(float **)buffer withBufferSize:(UInt32)bufferSize withNumberOfChannels:(UInt32)numberOfChannels {
    /*
     *  Returns a float array called buffer that contains the stereo signal data
     *  buffer[0] is the left audio channel
     *  buffer[1] is the right audio channel
     */

    // Using a separate audio thread to not block the main UI thread
    dispatch_async(dispatch_get_main_queue(), ^{

        float decibels = [self getDecibelsFromVolume:buffer withBufferSize:bufferSize];

        NSLog(@"Decibels: %f", decibels);

    });

}

问题

问题是,在通过以下链接实施解决方案后,我不明白它是如何工作的。如果有人能解释它如何将音量转换为分贝,我将不胜感激


编码

该解决方案使用Accelerate Framework中的以下方法将音量转换为分贝:

下面是getDecibelsFromVolume从 EZAudio Delegate 调用的方法。它是从委托中传递float** buffer过来的。bufferSize

- (float)getDecibelsFromVolume:(float**)buffer withBufferSize:(UInt32)bufferSize {

    // Decibel Calculation.

    float one = 1.0;
    float meanVal = 0.0;
    float tiny = 0.1;
    float lastdbValue = 0.0;

    vDSP_vsq(buffer[0], 1, buffer[0], 1, bufferSize);

    vDSP_meanv(buffer[0], 1, &meanVal, bufferSize);

    vDSP_vdbcon(&meanVal, 1, &one, &meanVal, 1, 1, 0);


    // Exponential moving average to dB level to only get continous sounds.

    float currentdb = 1.0 - (fabs(meanVal) / 100);

    if (lastdbValue == INFINITY || lastdbValue == -INFINITY || isnan(lastdbValue)) {
        lastdbValue = 0.0;
    }

    float dbValue = ((1.0 - tiny) * lastdbValue) + tiny * currentdb;

    lastdbValue = dbValue;

    return dbValue;
}
4

1 回答 1

14

我将解释如何使用代码计算信号的 dB 值,然后展示它与 vDSP 示例的关系。

首先,计算一块数据的 RMS 和

double sumSquared = 0;
for (int i = 0 ; i < numSamples ; i++)
{
   sumSquared += samples[i]*samples[i];
}
double rms = sumSquared/numSamples;

有关RMS的更多信息

接下来将 RMS 值转换为 dB

double dBvalue = 20*log10(rms);

这与示例代码有何关系

vDSP_vsq(buffer[0], 1, buffer[0], 1, bufferSize);

该行在缓冲区上循环并计算缓冲区中所有元素的平方。如果缓冲区[1,2,3,4]在调用之前包含值,那么在调用之后它将包含值[1,4,9,16]

vDSP_meanv(buffer[0], 1, &meanVal, bufferSize);

此行在缓冲区上循环,将缓冲区中的值相加,然后返回总和除以元素的数量。因此对于输入缓冲区[1,4,9,16]计算总和30,除以4并返回结果7.5

vDSP_vdbcon(&meanVal, 1, &one, &meanVal, 1, 1, 0);

此行将 转换meanVal为分贝。在这里调用向量化函数确实没有意义,因为它只对单个元素进行操作。然而,它正在做的是将参数插入以下公式:

meanVal = n*log10(meanVal/one)

其中n1020取决于最后一个参数。在这种情况下,它是1010用于功率测量并20用于幅度。我认为20您使用它会更有意义。

最后一点代码看起来正在对结果进行一些简单的平滑处理,以减少仪表的弹性。

于 2015-02-26T04:35:11.697 回答