MediaRecorder генерирует исключение при настройке источника звука (разработка Android)

Я пытаюсь создать простое приложение для записи голосовой команды и отправить его в Amazon Alexa, для этого я следую этому руководству для записи голосовой команды. Следующий фрагмент кода вызывает у меня проблемы:

import android.app.Activity;
import android.media.MediaPlayer;
import android.media.MediaRecorder;
import android.os.Bundle;
import android.os.Environment;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View;
import android.widget.Button;
import android.widget.Toast;

import java.io.IOException;


public class MainActivity extends Activity {
    Button play,stop,record;
    private MediaRecorder myAudioRecorder;
    private String outputFile = null;

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    play=(Button)findViewById(R.id.button3);
    stop=(Button)findViewById(R.id.button2);
    record=(Button)findViewById(R.id.button);

    stop.setEnabled(false);
    play.setEnabled(false);
    outputFile = Environment.getExternalStorageDirectory().getAbsolutePath() + "/recording.3gp";;

    myAudioRecorder=new MediaRecorder();
    myAudioRecorder.setAudioSource(MediaRecorder.AudioSource.MIC); ***ERROR***
    myAudioRecorder.setOutputFormat(MediaRecorder.OutputFormat.THREE_GPP);
    myAudioRecorder.setAudioEncoder(MediaRecorder.OutputFormat.AMR_NB);
    myAudioRecorder.setOutputFile(outputFile);

Кидая следующий стек ошибок:

FATAL EXCEPTION: main
                                                                          Process: com.example.michael.test, PID: 20504
                                                                          java.lang.RuntimeException: Unable to start activity ComponentInfo{com.example.michael.test/com.example.michael.test.MainActivity}: java.lang.RuntimeException: setAudioSource failed.
                                                                              at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2416)
                                                                              at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2476)
                                                                              at android.app.ActivityThread.-wrap11(ActivityThread.java)
                                                                              at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1344)
                                                                              at android.os.Handler.dispatchMessage(Handler.java:102)
                                                                              at android.os.Looper.loop(Looper.java:148)
                                                                              at android.app.ActivityThread.main(ActivityThread.java:5417)
                                                                              at java.lang.reflect.Method.invoke(Native Method)
                                                                              at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:726)
                                                                              at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:616)
                                                                           Caused by: java.lang.RuntimeException: setAudioSource failed.
                                                                              at android.media.MediaRecorder.setAudioSource(Native Method)
                                                                              at com.example.michael.test.MainActivity.onCreate(MainActivity.java:37)
                                                                              at android.app.Activity.performCreate(Activity.java:6251)
                                                                              at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1107)
                                                                              at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2369)
                                                                              at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2476) 
                                                                              at android.app.ActivityThread.-wrap11(ActivityThread.java) 
                                                                              at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1344) 
                                                                              at android.os.Handler.dispatchMessage(Handler.java:102) 
                                                                              at android.os.Looper.loop(Looper.java:148) 
                                                                              at android.app.ActivityThread.main(ActivityThread.java:5417) 
                                                                              at java.lang.reflect.Method.invoke(Native Method) 
                                                                              at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:726) 
                                                                              at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:616) 
04-25 12:35:39.109 20504-20504/com.example.michael.test I/Process: Sending signal. PID: 20504 SIG: 9

Все вопросы, связанные с поиском в Google, связаны с неправильными настройками прав доступа приложения к источнику звука, однако мой файл androidmanifest.xml содержит следующие строки:

    <uses-permission android:name="android.permission.MODIFY_AUDIO_SETTINGS" />
<uses-permission android:name="android.permission.RECORD_AUDIO" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.WRITE_SETTINGS" />

Так что я в недоумении, что еще может быть не так? Я разрабатываю для Android Studio, используя LG Nexus 5 с Android 6.0.1 для тестирования. SDK установлен на Android 5.0.

1 ответ

После трех дней исследований я решил проверить Диспетчер приложений ("Настройки"> "Приложения"> "Диспетчер приложений"> "MyRecorderApp"> "Разрешения") во время работы MyRecorderApp. Как оказалось, для MyRecorderApp не было "Опасных разрешений" (как их называет Google), поэтому я сначала включил их вручную (с тех пор мое приложение перестало зависать), что привело меня к поиску информации ниже (от Google и этого блога)).

Начиная с Android 6.0 (уровень API 23), пользователи предоставляют разрешения приложениям во время работы приложения, а не при установке приложения.

Цитата из Google: полное объяснение с сайта разработчиков Google

Ниже приведена краткая версия кода, которую я получил на странице Google и вышеупомянутых источниках. Идея состоит в том, чтобы запросить у пользователя разрешение на запись звука и доступ к папке телефона при первом запуске приложения, то есть после его установки. Это сработало для моего Android M (6.0.1), Android Studio 2.1.1

Добавьте следующие объявления

private boolean permissionToRecordAccepted = false;
private boolean permissionToWriteAccepted = false;
private String [] permissions = {"android.permission.RECORD_AUDIO", "android.permission.WRITE_EXTERNAL_STORAGE"};

Добавьте следующий блок кода переопределения. Вы также можете добавить его с помощью шаблона Android Studio, нажав "Alt" + "Вставить" (Windows) или "control" + "O" (Mac), а затем найдите параметр onRequestPermissionsResult, который будет внизу списка.

@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
    super.onRequestPermissionsResult(requestCode, permissions, grantResults);
    switch (requestCode){
        case 200:
            permissionToRecordAccepted  = grantResults[0] == PackageManager.PERMISSION_GRANTED;
            permissionToWriteAccepted  = grantResults[1] == PackageManager.PERMISSION_GRANTED;
            break;
    }
    if (!permissionToRecordAccepted ) MainActivity.super.finish();
    if (!permissionToWriteAccepted ) MainActivity.super.finish();

}

Наконец, добавьте этот оператор "if" в ваш onCreate.

@Override
protected void onCreate(Bundle savedInstanceState) {

    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    // Add the following code to your onCreate
    int requestCode = 200;
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
        requestPermissions(permissions, requestCode);
    }

}
Другие вопросы по тегам