Android AudioRecord 不会初始化

Android AudioRecord won't initialize

我正在尝试实现一个应用程序来侦听微phone 输入(特别是呼吸),并根据它呈现数据。我正在使用 Android class AudioRecord,在尝试实例化 AudioRecord 时出现三个错误。

AudioRecord: AudioFlinger could not create record track, status: -1
AudioRecord-JNI: Error creating AudioRecord instance: initialization check failed with status -1.
android.media.AudioRecord: Error code -20 when initializing native AudioRecord object.

我找到了这个很棒的帖子:AudioRecord object not initializing

我从已接受的答案中借用了尝试所有采样率、音频格式和通道配置的代码来尝试解决问题,但没有帮助,我在所有设置中都遇到了上述错误。根据线程中的一个答案,我还在几个地方添加了对 AudioRecord.release() 的调用,但没有任何区别。

这是我的代码:

import android.media.AudioFormat;
import android.media.AudioRecord;
import android.media.MediaRecorder;
import android.util.Log;

public class SoundMeter {

private AudioRecord ar = null;
private int minSize;
private static int[] mSampleRates = new int[] { 8000, 11025, 22050, 32000, 44100 };

public boolean start() {
    ar = findAudioRecord();
    if(ar != null){
        ar.startRecording();
        return true;
    }
    else{
        Log.e("SoundMeter", "ERROR, could not create audio recorder");
        return false;
    }
}

public void stop() {
    if (ar != null) {
        ar.stop();
        ar.release();
    }
}

public double getAmplitude() {
    short[] buffer = new short[minSize];
    ar.read(buffer, 0, minSize);
    int max = 0;
    for (short s : buffer)
    {
        if (Math.abs(s) > max)
        {
            max = Math.abs(s);
        }
    }
    return max;
}

public AudioRecord findAudioRecord() {
    for (int rate : mSampleRates) {
        for (short audioFormat : new short[] { AudioFormat.ENCODING_PCM_8BIT, AudioFormat.ENCODING_PCM_16BIT, AudioFormat.ENCODING_PCM_FLOAT }) {
            for (short channelConfig : new short[] { AudioFormat.CHANNEL_IN_MONO, AudioFormat.CHANNEL_IN_STEREO }) {
                try {
                    Log.d("SoundMeter", "Attempting rate " + rate + "Hz, bits: " + audioFormat + ", channel: " + channelConfig);
                    int bufferSize = AudioRecord.getMinBufferSize(rate, channelConfig, audioFormat);

                    if (bufferSize != AudioRecord.ERROR_BAD_VALUE) {
                        // check if we can instantiate and have a success
                        Log.d("SoundMeter", "Found a supported bufferSize, attempting to instantiate");
                        AudioRecord recorder = new AudioRecord(MediaRecorder.AudioSource.DEFAULT, rate, channelConfig, audioFormat, bufferSize);

                        if (recorder.getState() == AudioRecord.STATE_INITIALIZED){
                            minSize = bufferSize;
                            return recorder;
                        }
                        else
                            recorder.release();
                    }
                } catch (Exception e) {
                    Log.e("SoundMeter", rate + " Exception, keep trying.", e);
                }
            }
        }
    }
    return null;
}

}

我还添加了

<uses-permission android:name="android.permission.RECORD_AUDIO"/>

根据上述线程中的其他答案之一,将

标记添加到我的清单文件中,作为清单标记的子项和应用程序标记的兄弟项。我在添加这个标签后重建了项目。

这些是我在谷歌搜索问题时找到的解决方案,但不幸的是,它们似乎并不适合我。 我正在 Nexus 5 phone(不是模拟器)上进行调试。这些错误出现在调用 AudioRecord 的构造函数时。我已经多次重启 phone 以尝试释放 microphone,但无济于事。项目基于Android4.4,我的phone目前是运行Android6.0.1.

非常感谢关于我还可以尝试哪些我可能错过的一些提示。谢谢!

我自己找到了答案。它与权限有关。

问题是我 运行ning API 版本 23 (Android 6.0.1) 在我的 phone 上,它不再只使用清单文件来处理权限。从版本 23 开始,权限是在 运行 时间内授予的。我添加了一个确保在 运行 时间内请求权限的方法,当我在 phone 上允许它一次时,它起作用了。

private void requestRecordAudioPermission() {
    //check API version, do nothing if API version < 23!
    int currentapiVersion = android.os.Build.VERSION.SDK_INT;
    if (currentapiVersion > android.os.Build.VERSION_CODES.LOLLIPOP){

        if (ContextCompat.checkSelfPermission(this, Manifest.permission.RECORD_AUDIO) != PackageManager.PERMISSION_GRANTED) {

            // Should we show an explanation?
            if (ActivityCompat.shouldShowRequestPermissionRationale(this, Manifest.permission.RECORD_AUDIO)) {

                // Show an expanation to the user *asynchronously* -- don't block
                // this thread waiting for the user's response! After the user
                // sees the explanation, try again to request the permission.

            } else {

                // No explanation needed, we can request the permission.

                ActivityCompat.requestPermissions(this, new String[]{Manifest.permission.RECORD_AUDIO}, 1);
            }
        }
    }
}

@Override
public void onRequestPermissionsResult(int requestCode, String permissions[], int[] grantResults) {
    switch (requestCode) {
        case 1: {
            // If request is cancelled, the result arrays are empty.
            if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {

                // permission was granted, yay! Do the
                // contacts-related task you need to do.
                Log.d("Activity", "Granted!");

            } else {

                // permission denied, boo! Disable the
                // functionality that depends on this permission.
                Log.d("Activity", "Denied!");
                finish();
            }
            return;
        }

        // other 'case' lines to check for other
        // permissions this app might request
    }
}

然后,在创建 AudioRecord 之前,我从主 activity 中的 onCreate() 方法调用 requestRecordAudioPermission()。