###前提・実現したいこと
AndroidStudioでAudioRecordを利用して音声のデータ処理を行いたいのですが、集音がうまくいきません。
MainAvtivity.javaやEvent logにはエラーはなく実機でテストをしてみるとアプリは起動します。しかし集音を開始するボタンをタップするとアプリが落ちてしまいます。
またまずはAudioRecordを利用して録音だけをするアプリを作り試してみましたがそちらも似たようなエラーが表示されました。
完全に煮詰まっているので何か分かる方いましたら回答よろしくお願いします。
###発生している問題・エラーメッセージ
AndroidMonitorで実機のログを確認したところ java.lang.IllegalStateException: startRecording() called on an uninitialized AudioRecord. とあり初期化されていないAudioRecordが呼び出されてると表示されます。 AndroidMonitorのログでは アプリを起動すると図のように表示されます。集音を開始するとアプリが落ち、 10-20 05:34:16.391 25176-25176/jp.klab.sonic04 I/hwaps: JNI_OnLoad の下の部分が表示されます。 エミュレータでも実機でのテストの両方でうまくいきません。
###該当のソースコード
MainAvtivity.java import org.jtransforms.fft.DoubleFFT_1D; import android.media.AudioFormat; import android.media.AudioRecord; import android.media.MediaRecorder; import android.os.Bundle; import android.os.Handler; import android.os.Message; import android.support.v7.app.AppCompatActivity; import android.support.v7.widget.Toolbar; import android.util.Log; import android.view.View; import android.widget.Button; import android.widget.TextView; public class MainActivity extends AppCompatActivity implements Runnable, View.OnClickListener, Handler.Callback { private static final String TAG = "SNC"; private static final int SAMPLE_RATE = 44100; 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 mBufferSizeInShort; 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_main); 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); mBufferSizeInShort = bufferSizeInBytes / 2; // 録音用バッファ mRecordBuf = new short[mBufferSizeInShort]; // FFT 処理用 mFFTSize = mBufferSizeInShort; 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); } @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, mBufferSizeInShort); bSilence = true; for (int i = 0; i < mBufferSizeInShort; 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 / mFFTSize; } } AndroidManifest ... <uses-permission android:name="android.permission.RECORD_AUDIO"/> <application android:allowBackup="true" ...
###試したこと
実機を再起動などを試してみましたがうまくいきませんでした。
###補足情報(言語/FW/ツール等のバージョンなど)
AndroidStudio2.3.3
参考
https://github.com/mkttanabe/Sonic/tree/master/sonic04
こちらのGithubのデータを使用しています。
何かわかりましたら回答よろしくお願いします。
ご協力お願いします。
バッドをするには、ログインかつ
こちらの条件を満たす必要があります。