java - Android AudioRecord 不会初始化

标签 java android multithreading audiorecord

我正在尝试实现一个应用程序来监听麦克风输入(特别是呼吸),并根据它呈现数据。我正在使用 Android 类 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

我从已接受的答案中借用了尝试所有采样率、音频格式和 channel 配置的代码来尝试解决问题,但没有帮助,我在所有设置中都遇到了上述错误。根据线程中的一个答案,我还在几个地方添加了对 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"/>

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

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

这些是我在谷歌搜索问题时找到的解决方案,但不幸的是,它们似乎并不适合我。 我正在调试我的 Nexus 5 手机(不是模拟器)。这些错误出现在调用 AudioRecord 的构造函数时。我已多次重启手机以尝试释放麦克风,但无济于事。该项目基于Android 4.4,我的手机目前运行的是Android 6.0.1。

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

最佳答案

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

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

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()。

关于java - Android AudioRecord 不会初始化,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38033068/

相关文章:

java - 如何从 Testcontainer 的应用程序文件加载属性?

java - AndEngine - 根据父按钮是否被按下来启用禁用的子按钮

java - 如何在 Intellij IDEA 中使生成的方法位于类的末尾?

java - 尝试查看数据时,Azure Android CloudAppendBlob 导致错误 409(冲突)

multithreading - Delphi XE 中的辅助线程可以接收事件吗?

java - C++ 映射中的 java `TreeMap.lowerEntry` 和 `TreeMap.higherEntry` 等价吗?

android - Caused by : java. lang.UnsatisfiedLinkError: 无法加载库

android - 当x的值是连续的时,如何使用when(x){}?

multithreading - 如果我不知道将创建多少个线程,该如何等待所有线程?

java - ThreadPoolExecutor 没有正确收缩