我试图在Android中录制音频并将其存储在一个字节数组中,为此我使用了Android的AudioRecord内置类。
我之前已经使用过这门课了,一切都很好,但出于某种原因,似乎AudioRecord不再工作了。
问题在于AudioRecord被初始化并且没有显示任何错误,但是当它真正记录某些内容时,通过调用它的' startRecording()方法,有些东西失败了,我甚至没有得到正确的理由。
调用此方法后,这是 ALL Logcat的输出:
01-18 18:54:49.545 11303-11338/com.mypackage E/android.media.AudioRecord﹕ MediaRecorder prepare CallingPid = 11303 Callinguid= 10128
01-18 18:54:49.545 11303-11338/com.mypackage E/android.media.AudioRecord﹕ java.lang.Throwable
at android.media.AudioRecord.startRecording(AudioRecord.java:631)
at mypackage.AudioRecorder.record(AudioRecorder.java:179)
当然,在那之后没有录制音频,对于那些想知道它是否仅仅是内部错误消息的人来说。
这是初始化AudioRecord类的代码:
int bufferSize = AudioRecord.getMinBufferSize(AUDIO_SAMPLE_RATE, RECORDER_CHANNEL, RECORDER_AUDIO_ENCODING);
if (bufferSize != AudioRecord.ERROR_BAD_VALUE && bufferSize > 0)
{
// check if we can instantiate and have a success
AudioRecord recorder = new AudioRecord(MediaRecorder.AudioSource.MIC, AUDIO_SAMPLE_RATE, RECORDER_CHANNEL,
RECORDER_AUDIO_ENCODING, bufferSize);
if (recorder.getState() == AudioRecord.STATE_INITIALIZED)
{
m_Recorder = recorder;
m_RecordingThread = Executors.newSingleThreadExecutor();
m_IsInitialized = true;
return;
}
}
当使用的常量是:
/**
* Recorded audio's sample rate.
*/
private static final int AUDIO_SAMPLE_RATE = 44100;
/**
* Recorded audio's channel.
*/
private static final int RECORDER_CHANNEL = AudioFormat.CHANNEL_IN_MONO;
/**
* Recorded audio's encoding.
*/
private static final int RECORDER_AUDIO_ENCODING = AudioFormat.ENCODING_PCM_16BIT;
错误发生在这里:
if (m_Recorder != null)
m_Recorder.startRecording();
因此,在看到这种情况发生了几次后,我决定深入尝试使用Jetbrain的反编译器调试它,尽可能深地调试到AudioRecord类本身的错误源。
出于某种奇怪的原因,只要应用程序遇到位于错误源的断点,下一步就无法完成,因为AudioRecord再次崩溃,并出现相同的错误消息。
我的下一步是创建一个示例项目来测试AudioRecord的功能,但这次我还添加了一个AudioTrack,它会立即输出我的所有录音(一种环回),以防万一AudioRecord类设法记录某些内容,无论是否有错误
所以魔术已经发生了 - 应用程序正在运行,AudioRecord正在录制,而AudioTrack正在播放
AudioTrack类突然解决了我的所有问题,这似乎很奇怪,所以我从代码中删除了它并尝试再次运行应用程序。
这一次,AudioRecord没有遇到任何问题,即使我无法听到它的声音。输出,我可以向你保证它一直在录音。
稍微松了一口气,我想我自己终于克服了这个问题,我已经准备好向前推进,所以我添加了一些额外的代码来编码和解码录制的音频数据。
为了听取输出,我使用AudioTrack类返回了代码,并运行了应用程序。
想听一个有趣的事吗?该应用已崩溃并出现相同的错误刚刚。
现在理论上问题可能是在完成使用它时没有释放AudioRecord实例,但我从一开始就已经覆盖了它。
我试图在互联网上的任何地方寻找解决方案,但似乎以前没有人遇到过这样的问题。
另外,我想指出我已尝试重置设备,ADB插件以及重建所有项目。
我使用的是带有Android Lollipop(API 21)的魅族MX5手机。
答案 0 :(得分:1)
好吧,我解决了这个问题,或者至少解决了它的一部分问题,导致它现在正常工作。
事情是,在调用AudioRecord类的read()方法时,我试图读取比预分配缓冲区大小更多的字节。
虽然它没有解决这篇文章的主要问题,这是奇怪的错误信息和未捕获的异常,但它使得AudioRecord完美地工作,所以对我来说这个线程可以看作是回答和关闭的!
答案 1 :(得分:1)
完整代码
import java.io.IOException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress;
import java.net.UnknownHostException;
//import com.rdt.facerecord.R;
import android.app.Activity;
import android.media.AudioFormat;
import android.media.AudioRecord;
import android.media.MediaRecorder;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.TextView;
public class StreamD extends Activity {
private Button startButton,stopButton;
public byte[] buffer;
public static DatagramSocket socket;
private int port=8089;
AudioRecord recorder;
private int sampleRate = 16000 ; // 44100 for music
private int channelConfig = AudioFormat.CHANNEL_CONFIGURATION_MONO;
private int audioFormat = AudioFormat.ENCODING_PCM_16BIT;
int minBufSize = AudioRecord.getMinBufferSize(sampleRate, channelConfig, audioFormat);
private boolean status = true;
TextView prompt;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.stream);
prompt = (TextView)findViewById(R.id.textView1);
startButton = (Button) findViewById (R.id.start_bbb);
stopButton = (Button) findViewById (R.id.stop_bbb);
startButton.setOnClickListener (startListener);
stopButton.setOnClickListener (stopListener);
}
private final OnClickListener stopListener = new OnClickListener() {
@Override
public void onClick(View arg0) {
status = false;
recorder.release();
System.out.println("Recorder released");
}
};
private final OnClickListener startListener = new OnClickListener() {
@Override
public void onClick(View arg0) {
status = true;
startStreaming();
}
};
public void startStreaming() {
Thread streamThread = new Thread(new Runnable() {
@Override
public void run() {
try {
DatagramSocket socket = new DatagramSocket();
System.out.println("Socket Created");
byte[] buffer = new byte[minBufSize];
System.out.println("Buffer created of size " + minBufSize);
DatagramPacket packet;
final InetAddress destination = InetAddress.getByName("218.000.000.000");
System.out.println("Address retrieved");
recorder = new AudioRecord(MediaRecorder.AudioSource.MIC,sampleRate,channelConfig,audioFormat,m inBufSize*10);
System.out.println("Recorder initialized");
recorder.startRecording();
while(status == true) {
//reading data from MIC into buffer
minBufSize = recorder.read(buffer, 0, buffer.length);
//putting buffer in the packet
packet = new DatagramPacket (buffer,buffer.length,destination,port);
socket.send(packet);
System.out.println("MinBufferSize: " +minBufSize);
}
} catch(UnknownHostException e) {
System.out.println("UnknownHostException");
} catch (IOException e) {
e.printStackTrace();
System.out.println("IOException");
}
}
});
streamThread.start();
}
}
并添加权限
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.RECORD_AUDIO" />
答案 2 :(得分:0)
我在魅族 m2 上遇到了同样的错误信息。经过 7 个小时的努力寻找原因,我意识到这只是一个内部 Log.e (...)。在我看来,我的代码因为这个错误而无法工作,但实际上我不得不在一个完全不同的地方寻找错误。
我们应该忽略这个错误信息。