以下のコードでは、私のaudioRecord
オブジェクトは初期化されていません。 onCreate
メソッドに移動してみて、グローバルにしました。状態をログに記録したところ、1
の値が返されます。これは、すぐに使用できることを意味します。デバッガーは、startRecording
が初期化されていないオブジェクトで呼び出されていると言います。また、オーディオソースを取得できなかったということです。
これらのエラーが発生するのはなぜですか?
package com.tecmark;
import Java.io.BufferedOutputStream;
import Java.io.DataOutputStream;
import Java.io.File;
import Java.io.FileOutputStream;
import Android.app.Activity;
import Android.media.AudioFormat;
import Android.media.AudioRecord;
import Android.media.MediaRecorder;
import Android.os.Bundle;
import Android.os.Environment;
import Android.util.Log;
import Android.view.View;
import Android.widget.TextView;
public class recorder extends Activity {
private Thread thread;
private boolean isRecording;
private AudioRecord recorder;
private FileOutputStream os;
private BufferedOutputStream bos;
private DataOutputStream dos;
private TextView text;
private int audioSource = MediaRecorder.AudioSource.MIC;
private int sampleRate = 22050;
private int channel = AudioFormat.CHANNEL_CONFIGURATION_MONO;
private int encoding = AudioFormat.ENCODING_PCM_16BIT;
private int result = 0;
private int bufferSize;
private byte[] buffer;
/** Called when the activity is first created. */
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
Log.v("onCreate", "layout set, about to init audiorec obj");
text = (TextView)findViewById(R.id.TextView01);
bufferSize = AudioRecord.getMinBufferSize(sampleRate,channel,encoding);
buffer = new byte[bufferSize];
recorder = new AudioRecord(audioSource, sampleRate,channel,encoding,
AudioRecord.getMinBufferSize(sampleRate, channel,encoding));
Log.i("recorder obj state",""+recorder.getRecordingState());
}
public void onClickPlay(View v){
}
public void record(){
Log.i("inside record method", "******");
File path = Environment.getExternalStorageDirectory();
Log.v("file path", ""+path.getAbsolutePath());
File file = new File(path, "test.wav");
if(file.exists()){
file.delete();
}
path.mkdirs();
Log.v("file path", ""+file.getAbsolutePath());
try {
os = new FileOutputStream(file);
bos = new BufferedOutputStream(os);
dos = new DataOutputStream(bos);
} catch (Exception e1) {
e1.printStackTrace();
}
int bufferSize = AudioRecord.getMinBufferSize(sampleRate,channel,encoding);
byte[] buffer = new byte[bufferSize];
recorder.startRecording();
isRecording = true;
try{
while (isRecording){
result = recorder.read(buffer, 0, bufferSize);
for(int a=0; a<result;a++){
dos.write(buffer[a]);
if(!isRecording){
recorder.stop();
break;
}
}
}
dos.flush();
dos.close();
}catch(Exception e){
e.printStackTrace();
}
}// end of record method
public void onClickStop(View v){
Log.v("onClickStop", "stop clicked");
isRecording=false;
}
public void onClickReverse(View v){
Log.v("onClickReverse", "reverse clicked");
}
public void onClickRecord(View v){
Log.v("onClickRecourd", "record clicked, thread gona start");
text.setText("recording");
thread = new Thread(new Runnable() {
public void run() {
isRecording = true;
record();
}
});
thread.start();
isRecording = false;
}
}//end of class
Logcat
01-30 15:23:16.724: ERROR/AudioRecord(12817): Could not get audio input for record source 1 01-30 15:23:16.729:
ERROR/AudioRecord-JNI(12817): Error creating AudioRecord instance: initialization check failed. 01-30 15:23:16.729:
ERROR/AudioRecord-Java(12817): [ Android.media.AudioRecord ] Error code
-20 when initializing native AudioRecord object. 01-30 15:23:16.729: INFO/recorder obj state(12817): 1 01-30 15:23:16.729:
WARN/dalvikvm(12817): threadid=13: thread exiting with uncaught exception (group=0x4001b180) 01-30 15:23:16.729:
ERROR/AndroidRuntime(12817): Uncaught handler: thread Thread-7 exiting due to uncaught exception 01-30 15:23:16.739:
ERROR/AndroidRuntime(12817): Java.lang.IllegalStateException: startRecording() called on an uninitialized AudioRecord. 01-30 15:23:16.739:
ERROR/AndroidRuntime(12817): at Android.media.AudioRecord.startRecording(AudioRecord.Java:495) 01-30 15:23:16.739:
ERROR/AndroidRuntime(12817): at com.tecmark.recorder.record(recorder.Java:114) 01-30 15:23:16.739:
ERROR/AndroidRuntime(12817): at com.tecmark.recorder$1.run(recorder.Java:175) 01-30 15:23:16.739:
ERROR/AndroidRuntime(12817): at Java.lang.Thread.run(Thread.Java:1096)
AudioRecordを使用するコツは、各デバイスが異なる初期化設定を持つ可能性があるため、ビットレート、エンコーディングなどの考えられるすべての組み合わせをループするメソッドを作成する必要があることです。
private static int[] mSampleRates = new int[] { 8000, 11025, 22050, 44100 };
public AudioRecord findAudioRecord() {
for (int rate : mSampleRates) {
for (short audioFormat : new short[] { AudioFormat.ENCODING_PCM_8BIT, AudioFormat.ENCODING_PCM_16BIT }) {
for (short channelConfig : new short[] { AudioFormat.CHANNEL_IN_MONO, AudioFormat.CHANNEL_IN_STEREO }) {
try {
Log.d(C.TAG, "Attempting rate " + rate + "Hz, bits: " + audioFormat + ", channel: "
+ channelConfig);
int bufferSize = AudioRecord.getMinBufferSize(rate, channelConfig, audioFormat);
if (bufferSize != AudioRecord.ERROR_BAD_VALUE) {
// check if we can instantiate and have a success
AudioRecord recorder = new AudioRecord(AudioSource.DEFAULT, rate, channelConfig, audioFormat, bufferSize);
if (recorder.getState() == AudioRecord.STATE_INITIALIZED)
return recorder;
}
} catch (Exception e) {
Log.e(C.TAG, rate + "Exception, keep trying.",e);
}
}
}
}
return null;
}
AudioRecord recorder = findAudioRecord();
recorder.release();
私は同じ問題を抱えていました、それを置くことで解決しました
<uses-permission Android:name="Android.permission.RECORD_AUDIO"></uses-permission>
マニフェストに。
Lollipopから、ユーザーに各許可を具体的に尋ねる必要もあります。取り消された可能性があるため。許可が付与されていることを確認してください。
Javadocsによると、すべてのデバイスがこの形式をサポートすることが保証されています(記録用)。
44100、AudioFormat.CHANNEL_IN_MONO、AudioFormat.ENCODING_PCM_16BIT。
再生するには、CHANNEL_OUT_MONOに変更します。
上記の手順をすべて実行した後も同じ問題が発生していましたが、私にとって有効だったのは、私のosはMarshmallowであり、許可を求める必要がありましたでした。
次に、Lollipopを使用して、ユーザーに許可を個別に求める必要があります。許可が付与されていることを確認してください。
いくつかのAudioRecordオブジェクトの初期化に関する問題は、次のオブジェクトを作成する前にaudioRecord.release();
を使用することで修正できました...詳細: Android AudioRecord-2回目に初期化しない
お使いの携帯電話システムがAndroid M以上の場合、おそらくAndroid M . http:// developer .Android.com/guide/topics/security/permissions.html
私の場合、ショーン・ズーがコメントしたように、マイクのAndroid 7で許可を手動で許可する必要がありました。
ちょうど同じ問題がありました。解決策は、デバイスを再起動することでした。コードで遊んでいる間、明らかにオーディオデバイスがスタックする原因となるAudioRecordオブジェクトをリリースしませんでした。オーディオデバイスが機能したかどうかをテストするために、Google PlayからAudalyzerをダウンロードしました。
実行中のavdのSDCardがいっぱいになると、AudioRecordコンストラクターがnullを返すことに気付きました。 SDカードをクリアしようとしましたか?
これは、メインアクティビティを一時停止したことをスレッドが認識せず、レコーダーを停止した後も記録しようとしていることに関係していると思います。
OnResume()およびonPause()メソッドを変更してisRecordingブール値を変更することで解決しました。
_public void onResume() {
...
isRecording = true;
}
public void onPause() {
...
isRecording = false;
}
_
次に、スレッドで、isRecordingをチェックするifステートメントでstartRecording()
とstop()
の両方を囲みます。
_if(isRecording)
recorder.startRecording();
...
if(isRecording)
recorder.stop(); // which you've done
_
Xamarin Android AudioRecord with C#。
int[] sampleRates = new int[] { 44100, 22050, 11025, 8000 };
Encoding[] encodings = new Encoding[] { Encoding.Pcm8bit, Encoding.Pcm16bit };
ChannelIn[] channelConfigs = new ChannelIn[]{ ChannelIn.Mono, ChannelIn.Stereo };
//Not all of the formats are supported on each device
foreach (int sampleRate in sampleRates)
{
foreach (Encoding encoding in encodings)
{
foreach (ChannelIn channelConfig in channelConfigs)
{
try
{
Console.WriteLine("Attempting rate " + sampleRate + "Hz, bits: " + encoding + ", channel: " + channelConfig);
int bufferSize = AudioRecord.GetMinBufferSize(sampleRate, channelConfig, encoding);
if (bufferSize > 0)
{
// check if we can instantiate and have a success
AudioRecord recorder = new AudioRecord(AudioSource.Mic, sampleRate, channelConfig, encoding, bufferSize);
if (recorder.State == State.Initialized)
{
mBufferSize = bufferSize;
mSampleRate = sampleRate;
mChannelConfig = channelConfig;
mEncoding = encoding;
recorder.Release();
recorder = null;
return true;
}
}
}
catch (Exception ex)
{
Console.WriteLine(sampleRate + "Exception, keep trying." + ex.Message);
}
}
}
}