调用startRecording()方法

时间:2016-01-18 17:35:48

标签: java android audio logcat audiorecord

我试图在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手机。

3 个答案:

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

我们应该忽略这个错误信息。