我做正确的事情将分贝从-120 - 0转换为0 - 120

时间:2012-02-12 07:45:42

标签: iphone ios xcode audio-recording decibel

我想测量周围的音量,不太确定我做的是否合适。

我想创建一个范围为0(安静)到120(非常嘈杂)的VU表。

我获得了峰值和平均功率,但在正常安静的环境中非常高。 请给我一些指针。

- (void)viewDidLoad
{
    [super viewDidLoad];
    // Do any additional setup after loading the view, typically from a nib.


    //creating an audio CAF file in the temporary directory, this isn’t ideal but it’s the only way to get this class functioning (the temporary directory is erased once the app quits). Here we also specifying a sample rate of 44.1kHz (which is capable of representing 22 kHz of sound frequencies according to the Nyquist theorem), and 1 channel (we do not need stereo to measure noise).

    NSDictionary* recorderSettings = [NSDictionary dictionaryWithObjectsAndKeys:
                                      [NSNumber numberWithInt:kAudioFormatLinearPCM],AVFormatIDKey,
                                      [NSNumber numberWithInt:44100],AVSampleRateKey,
                                      [NSNumber numberWithInt:1],AVNumberOfChannelsKey,
                                      [NSNumber numberWithInt:16],AVLinearPCMBitDepthKey,
                                      [NSNumber numberWithBool:NO],AVLinearPCMIsBigEndianKey,
                                      [NSNumber numberWithBool:NO],AVLinearPCMIsFloatKey,
                                      nil];
    NSError* error;

    NSURL *url = [NSURL fileURLWithPath:@"/dev/null"];
    recorder = [[AVAudioRecorder alloc] initWithURL:url settings:recorderSettings error:&error];

    //enable measuring
    //tell the recorder to start recording:
    [recorder record];

    if (recorder) {
        [recorder prepareToRecord];
        recorder.meteringEnabled = YES;
        [recorder record];
        levelTimer = [NSTimer scheduledTimerWithTimeInterval: 0.01 target: self selector: @selector(levelTimerCallback:) userInfo: nil repeats: YES];

    } else
    {
        NSLog(@"%@",[error description]);
    }        
}

- (void)levelTimerCallback:(NSTimer *)timer {
    [recorder updateMeters];

    const double ALPHA = 0.05;
    double peakPowerForChannel = pow(10, (0.05 * [recorder averagePowerForChannel:0]));
    lowPassResults = ALPHA * peakPowerForChannel + (1.0 - ALPHA) * lowPassResults;     

    NSLog(@"Average input: %f Peak input: %f Low pass results: %f", [recorder averagePowerForChannel:0], [recorder peakPowerForChannel:0], lowPassResults);

    float tavgPow =[recorder averagePowerForChannel:0] + 120.0;
    float tpPow = [recorder peakPowerForChannel:0] + 120.0;

    float avgPow = tavgPow;//(float)abs([recorder averagePowerForChannel:0]);
    float pPow = tpPow;//(float)abs([recorder peakPowerForChannel:0]);

    NSString *tempAvg = [NSString stringWithFormat:@"%0.2f",avgPow];
        NSString *temppeak = [NSString stringWithFormat:@"%0.2f",pPow];
    [avg setText:tempAvg];
        [peak setText:temppeak];
    NSLog(@"Average input: %f Peak input: %f Low pass results: %f", avgPow,pPow , lowPassResults);
}   

5 个答案:

答案 0 :(得分:24)

Apple在其SpeakHere示例中使用查找表,该表格从dB转换为电平表上显示的线性值。这是为了节省设备功率(我猜)。

我也需要这个,但是并不认为每1/10(我的刷新率)进行几次浮点计算会花费很多设备功率。因此,我没有建立一个表格,而是将代码模塑成:

float       level;                // The linear 0.0 .. 1.0 value we need.
const float minDecibels = -80.0f; // Or use -60dB, which I measured in a silent room.
float       decibels    = [audioRecorder averagePowerForChannel:0];

if (decibels < minDecibels)
{
    level = 0.0f;
}
else if (decibels >= 0.0f)
{
    level = 1.0f;
}
else
{
    float   root            = 2.0f;
    float   minAmp          = powf(10.0f, 0.05f * minDecibels);
    float   inverseAmpRange = 1.0f / (1.0f - minAmp);
    float   amp             = powf(10.0f, 0.05f * decibels);
    float   adjAmp          = (amp - minAmp) * inverseAmpRange;

    level = powf(adjAmp, 1.0f / root);
}

我正在使用AVAudioRecorder,因此您会看到averagePowerForChannel:获得dB,但您可以在那里填写自己的dB值。

Apple的示例使用double计算,我不明白,因为音频计量float准确度绰绰有余,设备功耗更低。

毋庸置疑,您现在可以使用简单的level将此计算的level * 120.0f缩放到0 ... 120范围。

通过将root替换为2.0f,我们可以在powf(adjAmp, 1.0f / root)处修复sqrtf(adjAmp)时加快上述代码的速度;但这是一个小问题,一个非常好的编译器可能能够为我们这样做。我几乎肯定inverseAmpRange将在编译时计算一次。

答案 1 :(得分:8)

当您想使用1.0作为参考(0db)时,将线性幅度转换为分贝的公式是

20 * log10(amp);

所以我不确定查看代码的意图,但你可能想要

float db = 20 * log10([recorder averagePowerForChannel:0]);

这将从幅度为零的-infinity到幅度为1的0db。 如果你真的需要它在0到120之间,你可以添加120并在零处使用最大函数。

所以,在上面一行之后:

db += 120;
db = db < 0 ? 0 : db;

您使用的公式似乎是将DB转换为放大器的公式,我认为这与您想要的相反。

编辑:我重读了,看起来你可能已经有了分贝值。

如果是这种情况,请不要转换为幅度并添加120。

所以改变

double peakPowerForChannel = pow(10, (0.05 * [recorder averagePowerForChannel:0]));

double peakPowerForChannel = [recorder averagePowerForChannel:0];

你应该可以去。

答案 2 :(得分:1)

实际上,分贝的范围是从-160到0,但它可以转到正值。(AVAudioRecorder Class Reference - averagePowerForChannel:method)

然后最好写db += 160;而不是db += 120;。当然你也可以设置一个偏移来纠正它。

答案 3 :(得分:0)

我制作一个回归模型来转换从NSRecorder生成的wav数据与来自NSRecorder.averagePowerForChannel的分贝数据之间的映射关系

NSRecorder.averagePowerForChannel(dB)= -80 + 6 log2(wav_RMS

其中wav_RMS是短时间内wav数据的均方根值,即0.1秒。

答案 4 :(得分:-3)

只需设置最大值和最小值即可。就像你得到0-120的范围。如果你想要0-60的范围。只需将值除以一半即可获得半范围等等。