我正在查看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轴。
答案 0 :(得分:5)
我相信它使用的每个垃圾箱的频率由
给出yFract * hwSampleRate * .5
我很确定你需要.5因为yFract是总fftLength的一小部分,而FFT的最后一个bin对应于采样率的一半。因此,您可以执行类似
的操作NSLog(@"The magnitude for %f Hz is %f.", (yFract * hwSampleRate * .5), (interpVal * 120));
希望至少有助于指出正确的方向。