API < 26 没有虚拟方法 MediaRecorder.setOutputFile?

No virtual method MediaRecorder.setOutputFile with API < 26?

我正在我的应用程序中录制音频,但我在

中与 MediaRecorder class 作斗争
import android.media.MediaRecorder;

问题是代码可以编译,但在 API 级别 < 26 的设备上似乎 运行 没有,因为显然函数 MediaRecorder.setOutputFormat() 在 [= 之前​​不存在43=] 26 根据 Android Studio 中的内联错误消息。同样,当我 运行 模拟器上的代码 API 24 我遇到崩溃 运行 时间错误:

java.lang.NoSuchMethodError: No virtual method setOutputFile(Ljava/io/File;)V in class Landroid/media/MediaRecorder; or its super classes (declaration of 'android.media.MediaRecorder' appears in /system/framework/framework.jar)

作为快速修复,我只是将 gradle 文件中的最低 SDK 级别提高到 26,一切正常。但是,我不希望我的应用仅限于 Android 8.0 及更高版本,因为 7.x 仍然很受欢迎。

所以我的第一个问题是:怎么可能整个MediaRecorder库只有一个函数不可用,人家怎么用MediaRecorder 在 API 26 之前?我在网上找不到任何相关信息。

第二个问题:如何解决这个问题,以便我可以录制 API 24+ 级的音频?我必须使用其他图书馆吗?

这是我的 Activity 的一部分,我正在其中设置录音机并开始录音。同样,当最小 SDK 为 26 时,这工作正常...

private MediaRecorder recorder;

private void startRecording(){
    if(!recordingActive) {
        recorder = new MediaRecorder();
        recorder.setAudioSource(MediaRecorder.AudioSource.MIC);
        recorder.setAudioChannels(1);
        recorder.setOutputFormat(output_formats[currentFormat]);
        recorder.setAudioEncoder(MediaRecorder.AudioEncoder.AMR_NB);
        switch(activeAudioButton) {
            case AUDIO_BUTTON_A:
                recorder.setOutputFile(tempSoundFileA); // <-- 
                break;
            case AUDIO_BUTTON_B:
                recorder.setOutputFile(tempSoundFileB); // <-- 
                break;
        }
        recorder.setOnErrorListener(errorListener);
        recorder.setOnInfoListener(infoListener);
        try {
            recorder.prepare();
            recorder.start();
            recordingActive = true;
            runOnUiThread(new Runnable() {
                @Override
                public void run() {
                    speakNowTextView.setVisibility(View.VISIBLE);
                }
            });
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

这是我的 gradle 文件:

apply plugin: 'com.android.application'

android {
    compileSdkVersion 28
    defaultConfig {
        applicationId "com.my.app"
        minSdkVersion 24
        targetSdkVersion 28
        versionCode 1
        versionName "1.0"
        testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
    }
    buildTypes {
        release {
            minifyEnabled false
            proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
        }
    }
}

dependencies {
    implementation fileTree(dir: 'libs', include: ['*.jar'])
    implementation 'com.android.support:appcompat-v7:28.0.0'
    implementation 'com.android.support.constraint:constraint-layout:1.1.3'
    implementation 'com.android.support:support-v4:28.0.0'
    testImplementation 'junit:junit:4.12'
    androidTestImplementation 'com.android.support.test:runner:1.0.2'
    androidTestImplementation 'com.android.support.test.espresso:espresso-core:3.0.2'
    implementation 'com.android.support:recyclerview-v7:28.0.0'
    implementation 'com.android.support:design:28.0.0'
    implementation 'com.android.support:preference-v7:28.0.0'
    implementation 'com.android.support:recyclerview-v7:28.0.0'
}

setOutputFile 使用字符串作为参数而不是文件。

当使用 26 以下的 API 级别时,您可以传入包含文件路径的字符串。如果您特别需要使用 File 对象,请在 Java 中使用 file.getAbsolutePath() 或在 Kotlin 中使用 file.absolutePath