Android 调用 startRecording() 方法时 AudioRecord 失败
Android AudioRecord failing when calling the startRecording() method
我正在尝试在 android 中录制音频并将其存储在字节数组中,为此我使用 Android 的 AudioRecord 内置 class。
我以前已经用过这个 class,一切都很好,但由于某种原因,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 的代码class:
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 class 本身的错误源反编译器。
由于某些 st运行ge 原因,一旦应用程序遇到位于错误源的断点,下一步就无法完成,因为 AudioRecord 再次崩溃,并显示相同的错误消息.
我的下一步是创建一个示例项目来测试 AudioRecord 的功能,但这次我也添加了一个 AudioTrack,它将立即输出我的所有录音(一种回送),以防 AudioRecord class 设法记录了一些东西,有或没有错误。
神奇的事情发生了——应用程序正在运行,AudioRecord 正在录音,AudioTrack 正在播放。
AudioTrack class 突然解决了我所有的问题,这似乎真的很奇怪 运行ge,所以我将它从我的代码中删除并再次尝试 运行 该应用程序。
这一次,AudioRecord 的工作没有任何问题,即使我听不到它的输出,我可以向你保证它一直在录音。
我松了一口气,心想我终于克服了这个问题,我准备好继续前进了,所以我添加了一些额外的代码来编码和解码录制的音频数据。
为了听到输出,我使用 AudioTrack class 和应用程序 运行 返回了代码。
想听听有趣的事吗?该应用程序已崩溃并出现相同的错误 AGAIN。
现在理论上问题可能是在使用完 AudioRecord 实例后没有释放它,但我从一开始就已经介绍过了。
我已经尝试在 Internet 上几乎到处寻找解决方案,但似乎以前没有人遇到过这样的问题。
此外,我想指出我已经尝试重置我的设备、ADB 插件并重建所有项目。
我用的是魅族 MX5 phone,Android Lollipop(API 21).
好吧,我解决了问题,或者至少是部分问题,现在它可以正常工作了。
问题是在调用 AudioRecord class.
的 read() 方法时,我试图读取比预分配缓冲区大小更多的字节
虽然它没有解决这个 post 的主要问题,即奇怪的错误消息和未捕获的异常,但它使 AudioRecord 完美地工作,所以对我来说这个线程可以被视为已回答并关闭!
完整代码
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" />
我在魅族m2上遇到了同样的错误信息。经过 7 个小时的查找原因,我意识到这只是一个内部 Log.e (...)。在我看来,我的代码因为这个错误而无法运行,但实际上我不得不在一个完全不同的地方寻找错误。
我们应该忽略此错误消息。
我正在尝试在 android 中录制音频并将其存储在字节数组中,为此我使用 Android 的 AudioRecord 内置 class。
我以前已经用过这个 class,一切都很好,但由于某种原因,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 的代码class:
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 class 本身的错误源反编译器。
由于某些 st运行ge 原因,一旦应用程序遇到位于错误源的断点,下一步就无法完成,因为 AudioRecord 再次崩溃,并显示相同的错误消息.
我的下一步是创建一个示例项目来测试 AudioRecord 的功能,但这次我也添加了一个 AudioTrack,它将立即输出我的所有录音(一种回送),以防 AudioRecord class 设法记录了一些东西,有或没有错误。
神奇的事情发生了——应用程序正在运行,AudioRecord 正在录音,AudioTrack 正在播放。
AudioTrack class 突然解决了我所有的问题,这似乎真的很奇怪 运行ge,所以我将它从我的代码中删除并再次尝试 运行 该应用程序。
这一次,AudioRecord 的工作没有任何问题,即使我听不到它的输出,我可以向你保证它一直在录音。
我松了一口气,心想我终于克服了这个问题,我准备好继续前进了,所以我添加了一些额外的代码来编码和解码录制的音频数据。
为了听到输出,我使用 AudioTrack class 和应用程序 运行 返回了代码。
想听听有趣的事吗?该应用程序已崩溃并出现相同的错误 AGAIN。
现在理论上问题可能是在使用完 AudioRecord 实例后没有释放它,但我从一开始就已经介绍过了。
我已经尝试在 Internet 上几乎到处寻找解决方案,但似乎以前没有人遇到过这样的问题。
此外,我想指出我已经尝试重置我的设备、ADB 插件并重建所有项目。
我用的是魅族 MX5 phone,Android Lollipop(API 21).
好吧,我解决了问题,或者至少是部分问题,现在它可以正常工作了。
问题是在调用 AudioRecord class.
的 read() 方法时,我试图读取比预分配缓冲区大小更多的字节
虽然它没有解决这个 post 的主要问题,即奇怪的错误消息和未捕获的异常,但它使 AudioRecord 完美地工作,所以对我来说这个线程可以被视为已回答并关闭!
完整代码
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" />
我在魅族m2上遇到了同样的错误信息。经过 7 个小时的查找原因,我意识到这只是一个内部 Log.e (...)。在我看来,我的代码因为这个错误而无法运行,但实际上我不得不在一个完全不同的地方寻找错误。
我们应该忽略此错误消息。