###前提・実現したいこと
Android Studioで周波数検知を行うアプリを開発しているのですが、300HZを検知すると301HZで表示されるというように1や2HZのずれが生じてしまいます。
440HZや500HZで試した際にも1や2のずれが生じます。
この少しのずれを直したいです。また原因を知りたいです。
###作成したアプリケーションについて
アプリの内容自体は収音開始ボタンを押し、周波数のピークが表示されるといったものです。
###該当のソースコード
public class FFTActivity extends AppCompatActivity implements Runnable, View.OnClickListener, Handler.Callback { private static final String TAG = "SNC"; private static final int SAMPLE_RATE = 44100; private static final int FFT_SIZE = 4096; private static final short THRESHOLD_AMP = 0x00ff; private static final int MSG_RECORD_START = 100; private static final int MSG_RECORD_END = 110; private static final int MSG_FREQ_PEAK = 120; private static final int MSG_SILENCE = 130; private Handler mHandler; private AudioRecord mAudioRecord = null; private Button mButton01; private TextView mTextView02; private boolean mInRecording = false; private boolean mStop = false; private int mbufSize; private short mRecordBuf[]; private DoubleFFT_1D mFFT; private double mFFTBuffer[]; private int mFFTSize; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); Log.d(TAG, "onCreate"); mHandler = new Handler(this); setContentView(R.layout.activity_fft); Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar); setSupportActionBar(toolbar); mButton01 = (Button) findViewById(R.id.button01); mButton01.setOnClickListener(this); mTextView02 = (TextView) findViewById(R.id.textView02); int bufferSizeInBytes = AudioRecord.getMinBufferSize(SAMPLE_RATE, AudioFormat.CHANNEL_IN_MONO, AudioFormat.ENCODING_PCM_16BIT); System.out.println("FFT_SIZE" + FFT_SIZE); if (FFT_SIZE > mbufSize) mbufSize = FFT_SIZE; // 録音用バッファ mRecordBuf = new short[mbufSize]; // FFT 処理用 mFFTSize = mbufSize; mFFT = new DoubleFFT_1D(mFFTSize); mFFTBuffer = new double[mFFTSize]; mAudioRecord = new AudioRecord(MediaRecorder.AudioSource.MIC, SAMPLE_RATE, AudioFormat.CHANNEL_IN_MONO, AudioFormat.ENCODING_PCM_16BIT, bufferSizeInBytes); } //画面移動 public void onButton007( View v){ Intent intent = new Intent(this,Main2Activity.class); // 画面指定 startActivity(intent); // 画面を開く } @Override public void onStop() { super.onStop(); Log.d(TAG, "onStop"); } @Override public void onDestroy() { super.onDestroy(); Log.d(TAG, "onDestroy"); mStop = true; try { Thread.sleep(2000); } catch (InterruptedException e) { } if (mAudioRecord != null) { if (mAudioRecord.getRecordingState() != AudioRecord.RECORDSTATE_STOPPED) { Log.d(TAG, "cleanup mAudioRecord"); mAudioRecord.stop(); } mAudioRecord = null; } } @Override public void onClick(View v) { if (v == (View)mButton01) { // 集音開始 or 終了 if (!mInRecording) { mInRecording = true; new Thread(this).start(); } else { mInRecording = false; } } return; } @Override public boolean handleMessage(Message msg) { switch (msg.what) { case MSG_RECORD_START: Log.d(TAG, "MSG_RECORD_START"); mButton01.setText("STOP"); break; case MSG_RECORD_END: Log.d(TAG, "MSG_RECORD_END"); mButton01.setText("START"); break; case MSG_FREQ_PEAK: mTextView02.setText(Integer.toString(msg.arg1) + " Hz"); break; case MSG_SILENCE: mTextView02.setText(""); break; } return true; } @Override public void run() { boolean bSilence = false; mHandler.sendEmptyMessage(MSG_RECORD_START); // 集音開始 mAudioRecord.startRecording(); while (mInRecording && !mStop) { mAudioRecord.read(mRecordBuf, 0, mbufSize); bSilence = true; for (int i = 0; i < mbufSize; i++) { short s = mRecordBuf[i]; if (s > THRESHOLD_AMP) { bSilence = false; } } if (bSilence) { // 静寂 mHandler.sendEmptyMessage(MSG_SILENCE); continue; } int freq = doFFT(mRecordBuf); Message msg = new Message(); msg.what = MSG_FREQ_PEAK; msg.arg1 = freq; mHandler.sendMessage(msg); } // 集音終了 mAudioRecord.stop(); mHandler.sendEmptyMessage(MSG_RECORD_END); } private int doFFT(short[] data) { for (int i = 0; i < mFFTSize; i++) { mFFTBuffer[i] = (double)data[i]; } // FFT 実行 mFFT.realForward(mFFTBuffer); // 処理結果の複素数配列から各周波数成分の振幅値を求めピーク分の要素番号を得る double maxAmp = 0; int index = 0; for (int i = 0; i < mFFTSize/2; i++) { double a = mFFTBuffer[i*2]; // 実部 double b = mFFTBuffer[i*2 + 1]; // 虚部 // a+ib の絶対値 √ a^2 + b^2 = r が振幅値 double r = Math.sqrt(a*a + b*b); if (r > maxAmp) { maxAmp = r; index = i; } } // 要素番号・サンプリングレート・FFT サイズからピーク周波数を求める return index * SAMPLE_RATE / FFT_SIZE; } }
###試したこと
FFTサイズやバッファサイズを変更したりと試してみました。
よろしくお願いします。
###補足情報(言語/FW/ツール等のバージョンなど)
android studio

回答3件
あなたの回答
tips
プレビュー
バッドをするには、ログインかつ
こちらの条件を満たす必要があります。
2018/01/15 16:35