我已经在Android项目上工作了一段时间,该项目显示输入信号的基础频率(用作调音器)。我已成功实现了AudioRecord类,并从中获取数据。然而,我在对这些数据执行FFT以获取输入信号的基础频率方面遇到了困难。我一直在查看这里的帖子,并使用Java中的FFT和复数类。
我已成功使用Java中FFT中找到的FFT函数,但我不确定是否获得了正确的结果。对于FFT的幅度(sqrt[rere+imim]),我得到的值开始很高,约为15000 Hz,然后慢慢减少到约300 Hz。这似乎不对。
另外,就麦克风的原始数据而言,数据似乎很好,除了前50个左右的值始终是数字3,除非我在应用程序中仍然按下调音钮,那么我只会得到大约15个。这正常吗?
以下是我的部分代码。
首先,我使用来自我一直在查看的帖子的以下代码将从麦克风获取的短数据转换为double。我不完全理解这段代码,但我认为它有效。
代码接下来的内容如下:
那最后一段只是为了检查我得到的值(并保持理智!)。在上面提到的文章中,它谈到需要采样频率,并给出了以下代码:
我已成功使用Java中FFT中找到的FFT函数,但我不确定是否获得了正确的结果。对于FFT的幅度(sqrt[rere+imim]),我得到的值开始很高,约为15000 Hz,然后慢慢减少到约300 Hz。这似乎不对。
另外,就麦克风的原始数据而言,数据似乎很好,除了前50个左右的值始终是数字3,除非我在应用程序中仍然按下调音钮,那么我只会得到大约15个。这正常吗?
以下是我的部分代码。
首先,我使用来自我一直在查看的帖子的以下代码将从麦克风获取的短数据转换为double。我不完全理解这段代码,但我认为它有效。
//Conversion from short to double
double[] micBufferData = new double[bufferSizeInBytes];//size may need to change
final int bytesPerSample = 2; // As it is 16bit PCM
final double amplification = 1.0; // choose a number as you like
for (int index = 0, floatIndex = 0; index < bufferSizeInBytes - bytesPerSample + 1; index += bytesPerSample, floatIndex++) {
double sample = 0;
for (int b = 0; b < bytesPerSample; b++) {
int v = audioData[index + b];
if (b < bytesPerSample - 1 || bytesPerSample == 1) {
v &= 0xFF;
}
sample += v << (b * 8);
}
double sample32 = amplification * (sample / 32768.0);
micBufferData[floatIndex] = sample32;
}
代码接下来的内容如下:
//Create Complex array for use in FFT
Complex[] fftTempArray = new Complex[bufferSizeInBytes];
for (int i=0; i<bufferSizeInBytes; i++)
{
fftTempArray[i] = new Complex(micBufferData[i], 0);
}
//Obtain array of FFT data
final Complex[] fftArray = FFT.fft(fftTempArray);
final Complex[] fftInverse = FFT.ifft(fftTempArray);
//Create an array of magnitude of fftArray
double[] magnitude = new double[fftArray.length];
for (int i=0; i<fftArray.length; i++){
magnitude[i]= fftArray[i].abs();
}
fft.setTextColor(Color.GREEN);
fft.setText("fftArray is "+ fftArray[500] +" and fftTempArray is "+fftTempArray[500] + " and fftInverse is "+fftInverse[500]+" and audioData is "+audioData[500]+ " and magnitude is "+ magnitude[1] + ", "+magnitude[500]+", "+magnitude[1000]+" Good job!");
for(int i = 2; i < samples; i++){
fft.append(" " + magnitude[i] + " Hz");
}
那最后一段只是为了检查我得到的值(并保持理智!)。在上面提到的文章中,它谈到需要采样频率,并给出了以下代码:
private double ComputeFrequency(int arrayIndex) {
return ((1.0 * sampleRate) / (1.0 * fftOutWindowSize)) * arrayIndex;
}
我该如何实现这段代码?我不太明白fftOutWindowSize和arrayIndex是从哪里来的?
非常感谢您的帮助!
Dustin