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 (...)。在我看来,我的代码因为这个错误而无法运行,但实际上我不得不在一个完全不同的地方寻找错误。

我们应该忽略此错误消息。