在aurioTouch示例应用程序中确定哪些频率对应于x轴

时间:2010-06-01 12:59:26

标签: iphone audio fft

我正在查看iPhone SDK的aurioTouch sample application。当您选择“FFT”选项时,它已实现基本spectrum analyzer。应用程序缺少的一件事是X轴标签(即频率标签)。

aurioTouchAppDelegate.mm文件中,在第652行的函数- (void)drawOscilloscope中,它具有以下代码:

if (displayMode == aurioTouchDisplayModeOscilloscopeFFT)
{           
    if (fftBufferManager->HasNewAudioData())
    {
        if (fftBufferManager->ComputeFFT(l_fftData))
            [self setFFTData:l_fftData length:fftBufferManager->GetNumberFrames() / 2];
        else
            hasNewFFTData = NO;
    }

    if (hasNewFFTData)
    {

        int y, maxY;
        maxY = drawBufferLen;
        for (y=0; y<maxY; y++)
        {
            CGFloat yFract = (CGFloat)y / (CGFloat)(maxY - 1);
            CGFloat fftIdx = yFract * ((CGFloat)fftLength);

            double fftIdx_i, fftIdx_f;
            fftIdx_f = modf(fftIdx, &fftIdx_i);

            SInt8 fft_l, fft_r;
            CGFloat fft_l_fl, fft_r_fl;
            CGFloat interpVal;

            fft_l = (fftData[(int)fftIdx_i] & 0xFF000000) >> 24;
            fft_r = (fftData[(int)fftIdx_i + 1] & 0xFF000000) >> 24;
            fft_l_fl = (CGFloat)(fft_l + 80) / 64.;
            fft_r_fl = (CGFloat)(fft_r + 80) / 64.;
            interpVal = fft_l_fl * (1. - fftIdx_f) + fft_r_fl * fftIdx_f;

            interpVal = CLAMP(0., interpVal, 1.);

            drawBuffers[0][y] = (interpVal * 120);

        }
        cycleOscilloscopeLines();

    }

}

根据我的理解,这部分代码用于决定在UI中为每个频率绘制哪个幅度。我的问题是如何确定每次迭代(或y值)在for循环内表示的频率。

例如,如果我想知道6kHz的幅度是多少,我想添加一条类似于以下的线:

if (yValueRepresentskHz(y, 6))
  NSLog(@"The magnitude for 6kHz is %f", (interpVal * 120));

请注意,虽然他们选择使用变量名y,但根据我的理解,它实际上代表了频谱分析仪可视图中的x轴,以及{的值。 {1}}表示y轴。

1 个答案:

答案 0 :(得分:5)

我相信它使用的每个垃圾箱的频率由

给出
yFract * hwSampleRate * .5

我很确定你需要.5因为yFract是总fftLength的一小部分,而FFT的最后一个bin对应于采样率的一半。因此,您可以执行类似

的操作
NSLog(@"The magnitude for %f Hz is %f.", (yFract * hwSampleRate * .5), (interpVal * 120));

希望至少有助于指出正确的方向。