MediaRecorder не сохраняет звук в файл

Я пробую пример захвата звука Android (https://developer.android.com/guide/topics/media/audio-capture.html), но, похоже, он не работает на Samsung Galaxy S5 (единственный телефон, на котором я это тестировал). Это использует уровень API 23.

Аудиофайл создается на диске, но это 0-байтовый файл, что явно неверно. Это заставляет меня думать, что с MediaRecorder что-то не так.

Еще одно замечание: похоже, что getMaxAmplitude MediaRecorder работает - значит, у него есть доступ к микрофону.

Я нашел кучу других вопросов по SO, но ни на один из них нет ответа. Кто-нибудь сталкивался с этим в последнее время?


person Utkarsh Sinha    schedule 20.05.2016    source источник
comment
у вас есть необходимое разрешение?   -  person Konstantin    schedule 20.05.2016
comment
У меня включены права на запись звука и доступ в Интернет. @Ironman попробует сегодня.   -  person Utkarsh Sinha    schedule 20.05.2016


Ответы (3)


Я опубликовал полный код со всем этим, так что у вас нет проблем с этим.

activity_main.xml

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    android:paddingBottom="@dimen/activity_vertical_margin"
    tools:context=".MainActivity">

    <Button
        android:id="@+id/button1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_alignParentLeft="true"
        android:layout_alignParentTop="true"
        android:layout_marginLeft="68dp"
        android:layout_marginTop="50dp"
        android:text="Start Recording"
        />

    <Button
        android:id="@+id/button2"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_alignLeft="@+id/button1"
        android:layout_below="@+id/button1"
        android:layout_marginTop="64dp"
        android:text="Stop Recording"
        />
</RelativeLayout>

MainActivity.java

public class MainActivity extends Activity implements View.OnClickListener{


    private Button startButton;
    private Button stopButton;
    private MediaRecorder mediaRecorder;
    private File audioFile;

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

        startButton = (Button) findViewById(R.id.button1);
        startButton.setOnClickListener(this);
        startButton.setText("start");

        stopButton = (Button) findViewById(R.id.button2);
        stopButton.setOnClickListener(this);
        stopButton.setEnabled(false);
        stopButton.setText("stop");

        audioFile = new File(Environment.getExternalStorageDirectory(),
                "audio_test4.3gp");

    }

    private void resetRecorder() {
        mediaRecorder.setAudioSource(MediaRecorder.AudioSource.MIC);
        mediaRecorder.setOutputFormat(MediaRecorder.OutputFormat.THREE_GPP);
        mediaRecorder.setAudioEncoder(MediaRecorder.AudioEncoder.AMR_NB);
        mediaRecorder.setAudioEncodingBitRate(16);
        mediaRecorder.setAudioSamplingRate(44100);
        mediaRecorder.setOutputFile(audioFile.getAbsolutePath());

        try {
            mediaRecorder.prepare();
        } catch (IllegalStateException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    @Override
    public void onClick(View v) {
        switch (v.getId()) {
            case R.id.button1:
                mediaRecorder = new MediaRecorder();
                resetRecorder();
                mediaRecorder.start();

                startButton.setEnabled(false);
                stopButton.setEnabled(true);
                break;
            case R.id.button2:
                try {
                    mediaRecorder.stop();
                }catch (RuntimeException ex){

                }
                mediaRecorder.release();
                mediaRecorder = null;

                startButton.setEnabled(true);
                stopButton.setEnabled(false);
                break;
        }
    }

    @Override
    protected void onPause() {
        super.onPause();

        if (mediaRecorder != null) {
            mediaRecorder.stop();
            mediaRecorder.release();
            mediaRecorder = null;
        }
    }


}

manifest.xml

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.softeng.audiorecording" >

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

    <application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme" >
        <activity
            android:name=".MainActivity"
            android:label="@string/app_name" >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
    </application>

</manifest>

Снимок экрана с монитора устройства Android:

введите здесь описание изображения

В моем java-коде я дал ему имя audio_test4.3gp, а в ScreenShot есть файл с тем же именем и размером 3104.

person Harshad Pansuriya    schedule 20.05.2016
comment
Проблема заключалась в том, что я не запускал prepare прямо перед start. Я надеялся подготовить MediaRecorder гораздо раньше, чем на самом деле записывать звук. Перемещение prepare и start рядом друг с другом, кажется, заставляет это работать. - person Utkarsh Sinha; 20.05.2016
comment
@UtkarshSinha рад вам помочь. - person Harshad Pansuriya; 21.05.2016
comment
@HarshadPansuriya Эта ошибка возникла при попытке запустить его. java.lang.IllegalStateException в android.media.MediaRecorder.start (собственный метод), пожалуйста, помогите мне решить эту проблему .. - person Pratiksha Shaha; 22.09.2020

Я предполагаю, что вы используете мобильный телефон, а не эмулятор.

Если вы попытаетесь найти свой файл сразу после записи с помощью Проводника файлов на своем компьютере, вы не увидите его на некоторых мобильных телефонах или не увидите его с нулевыми байтами.

Используйте стандартный файловый менеджер на своем телефоне, чтобы найти файл, и я уверен, что вы его найдете (если у вас нет некоторых исключений).

Удачи.

person user2396640    schedule 03.09.2017

Здесь у меня есть полный рабочий код для MediaRecorder, попробуйте это

    //For Creating new File everytime
    private static int audioIndex = 0;

 public void startRecording()
 {
    ++audioIndex;
    Log.d("Index", audioIndex + "");
    Ask.on(RecordingActivity.this)
            .forPermissions(Manifest.permission.WRITE_EXTERNAL_STORAGE, Manifest.permission.RECORD_AUDIO)
            .withRationales("Give Permissions")
            .when(new Ask.Permission() {
                @Override
                public void granted(List<String> permissions) {


                    PackageManager pmanager = RecordingActivity.this.getPackageManager();
                    if (pmanager.hasSystemFeature(PackageManager.FEATURE_MICROPHONE)) {


                        mFileName = Environment.getExternalStorageDirectory().getAbsolutePath();
                        mFileName += "/eatthatfrog" + audioIndex + ".3gp";
                        Log.d("File", mFileName);
                        mediaRecorder = new MediaRecorder();
                        mediaRecorder.setAudioSource(MediaRecorder.AudioSource.MIC);
                        mediaRecorder.setOutputFormat(MediaRecorder.OutputFormat.THREE_GPP);
                        mediaRecorder.setAudioEncoder(MediaRecorder.AudioEncoder.AMR_NB);
                        mediaRecorder.setOutputFile(mFileName);
                        try {
                            mediaRecorder.prepare();
                        } catch (IOException e) {
                            e.printStackTrace();
                        }
                        mediaRecorder.start();


                        Log.d("Recording ", "True");

                    } else {
                        Toast.makeText(RecordingActivity.this, "This device doesn't have a mic!", Toast.LENGTH_LONG).show();
                    }
                }

                @Override
                public void denied(List<String> permissions) {
                    Log.d("Permission Denied", permissions.toString());
                }
            }).go();
}


@Override
public void onClick(View v) {
    switch (v.getId())
    {
        case R.id.stop:
            mediaRecorder.stop();
            mediaRecorder.reset();
            mediaRecorder.release();
            myChronometer.stop();
    }

Использование compile 'com.vistrav: ask: 1.2' для разрешений времени выполнения

person Veeresh Charantimath    schedule 20.05.2016