gpt4 book ai didi

android - 更改 block 大小会导致 FFT 分析失败

转载 作者:行者123 更新时间:2023-11-30 01:34:30 25 4
gpt4 key购买 nike

我正在尝试录制音频并获取频率。我可以使用 44100 的采样率和 2048 的 block 大小成功地做到这一点。我相信 bin 大小约为 20。但是,如果我尝试将 block 大小增加到 4096,那么我不会得到准确的频率,只会得到同样不准确的频率,没有幅度/分贝。

我的录音任务如下:

 private class RecordAudio extends AsyncTask<Void, double[], Boolean> {

@Override
protected Boolean doInBackground(Void... params) {

int bufferSize = AudioRecord.getMinBufferSize(frequency,
channelConfiguration, audioEncoding);
audioRecord = new AudioRecord(
MediaRecorder.AudioSource.DEFAULT, frequency,
channelConfiguration, audioEncoding, bufferSize);
int bufferReadResult;
short[] buffer = new short[blockSize];
double[] toTransform = new double[blockSize];
try {
audioRecord.startRecording();
} catch (IllegalStateException e) {
Log.e("Recording failed", e.toString());

}
while (started) {
if (isCancelled() || (CANCELLED_FLAG == true)) {

started = false;
//publishProgress(cancelledResult);
Log.d("doInBackground", "Cancelling the RecordTask");
break;
} else {
bufferReadResult = audioRecord.read(buffer, 0, blockSize);

for (int i = 0; i < blockSize && i < bufferReadResult; i++) {
toTransform[i] = (double) buffer[i] / 32768.0; // signed 16 bit
}

transformer.ft(toTransform);

publishProgress(toTransform);

}

}
return true;
}
@Override
protected void onProgressUpdate(double[]...progress) {

int mPeakPos = 0;
double mMaxFFTSample = 150.0;
for (int i = 100; i < progress[0].length; i++) {
int x = i;
int downy = (int) (150 - (progress[0][i] * 10));
int upy = 150;
//Log.i("SETTT", "X: " + i + " downy: " + downy + " upy: " + upy);

if(downy < mMaxFFTSample)
{
mMaxFFTSample = downy;
mMag = mMaxFFTSample;
mPeakPos = i;
}
}

mFreq = (((1.0 * frequency) / (1.0 * blockSize)) * mPeakPos)/2;
//Log.i("SSS", "F: " + mFreq + " / " + "M: " + mMag);

Log.i("SETTT", "FREQ: " + mFreq + " MAG: " + mMaxFFTSample);

}
@Override
protected void onPostExecute(Boolean result) {
super.onPostExecute(result);
try{
audioRecord.stop();
}
catch(IllegalStateException e){
Log.e("Stop failed", e.toString());

}
}
}

希望有一个我所缺少的快速修复。谢谢。

最佳答案

您需要仔细查看 RealDoubleFft.ft 函数的文档。进入函数的值是实数,但输出的值是复数 FFT 系数,因此 toTransform[0] 是第一个系数 toTransform[1] 的实部是第一个系数的虚部,依此类推。最终的数组大小相同,但由于每个复数占用 2 个 double ,因此总共有 N/2 个系数,其中最后一个是 sampleRate/2 的系数。

接下来,由于您对大小感兴趣,因此需要计算复数的大小。对于复数 x = a + bj,大小 |x| = sqrt(a*a + b*b)

    double maxMag = 0;
int peakIndex = 0;
for (int i = 0; i < progress[0].length/2; i++)
{
double re = progress[i*2];
double im = progress[i*2+1];
double mag = Math.sqrt(re*re + im*im);
if (mag > maxMag)
{
peakIndex = i;
maxMag = mag;
}
}

double peakFreq = sampleRate/fftLen * i/2; // might need a bit of tweaking.
double magInDb = 20*Math.log10(mag);

关于android - 更改 block 大小会导致 FFT 分析失败,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35276094/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com