录音代码写完之后,开始测试
测试录音步骤:
步骤 | 结果 | 原因 |
---|---|---|
第一次启动APP 获取权限 | 第一次录音失败 | crash之后,录音通道被释放 |
获取权限(定位?录音权限 | 第二次录音正常 |
crash相关信息:发现第一次调用MediaRecorder.start(),出现IllegalStateException。
具体Log如下:
以上信息不能准确的发现出现问题的根本原因。
排除问题:
1.权限是否打开
AndroidManifest.xml 定义权限
checkPerMission
2.是否正常初始化:
mMediaRecorder.setAudioSource(MediaRecorder.AudioSource.MIC);// 设置麦克风
mMediaRecorder.setOutputFormat(MediaRecorder.OutputFormat.DEFAULT);
mMediaRecorder.setAudioEncoder(MediaRecorder.AudioEncoder.AMR_NB);
mMediaRecorder.setOutputFile(mFilePath);
mMediaRecorder.setMaxDuration(MAX_LENGTH);
mMediaRecorder.prepare();
mMediaRecorder.start();
3.资源是否正常释放:
mMediaRecorder.setOnErrorListener(null);
mMediaRecorder.setOnInfoListener(null);
mMediaRecorder.setPreviewDisplay(null);
mMediaRecorder.stop();
mMediaRecorder.reset();
mMediaRecorder.release();
mMediaRecorder = null;
4.打印更多的log,比如start failed:-xx类型的log?
01-21 16:08:28.675 E/MediaRecorder( 4722): start failed: -38
查阅资源发现,-38代表麦克风通道被占用。
通道占用判断方法
private boolean validateMicAvailability(){
Boolean available = true;
AudioRecord recorder =
new AudioRecord(MediaRecorder.AudioSource.MIC, 44100,
AudioFormat.CHANNEL_IN_MONO,
AudioFormat.ENCODING_DEFAULT, 44100);
try{
if(recorder.getRecordingState() != AudioRecord.RECORDSTATE_STOPPED ){
available = false;
}
recorder.startRecording();
if(recorder.getRecordingState() != AudioRecord.RECORDSTATE_RECORDING){
recorder.stop();
available = false;
}
recorder.stop();
} finally{
recorder.release();
recorder = null;
}
return available;
}
返回true就是没有被占用。
返回false就是被占用。
调用完validateMicAvailability方法之后,Micro通道已经被释放,其他模块即可以使用麦克风。
根本原因:
1.由于项目中还引用了科大讯飞离线版语音模块,语音模块初始化成功之后,离线唤醒功能开启之后,默认打开语音权限,即占用语音通道。所以导致录音功能无法正常使用。
2.如果项目既想录音功能存在,又想使用语音模块,那么同一时刻只能使用一个。比如当需要录音时,主动关闭语音模块,当录音完成之后,主动初始化语音模块。