Не найдена реализация для нативного кода в Android - PullRequest
0 голосов
/ 02 июля 2018

Теперь я использую Java-библиотеку ffmpeg в Android. Я хочу использовать его непосредственно в своем приложении, поэтому я добавляю эту библиотеку. Но у меня есть проблема с этим.

E/AndroidRuntime: FATAL EXCEPTION: main
              Process: com.softcode.kihnoplay, PID: 15313
              java.lang.UnsatisfiedLinkError: No implementation found for java.lang.String androidFfmpeg.ArmArchHelper.cpuArchFromJNI() (tried Java_androidFfmpeg_ArmArchHelper_cpuArchFromJNI and Java_androidFfmpeg_ArmArchHelper_cpuArchFromJNI__)
                  at androidFfmpeg.ArmArchHelper.cpuArchFromJNI(Native Method)
                  at androidFfmpeg.CpuArchHelper.getCpuArch(CpuArchHelper.java:16)
                  at androidFfmpeg.FFmpeg.loadBinary(FFmpeg.java:40)
                  at com.softcode.tablet.Mp3Concat_Thread.startThread(Mp3Concat_Thread.java:90)
                  at com.softcode.tablet.Mp3Concat_Thread.<init>(Mp3Concat_Thread.java:54)
                  at com.softcode.phone.record.view.RecordActivity.onCreate(RecordActivity.java:182)
                  at android.app.Activity.performCreate(Activity.java:6955)
                  at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1126)
                  at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2927)
                  at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:3045)
                  at android.app.ActivityThread.-wrap14(ActivityThread.java)
                  at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1642)
                  at android.os.Handler.dispatchMessage(Handler.java:102)
                  at android.os.Looper.loop(Looper.java:154)
                  at android.app.ActivityThread.main(ActivityThread.java:6776)
                  at java.lang.reflect.Method.invoke(Native Method)
                  at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1520)
                  at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1410)

Я не знаю, почему происходит эта ошибка.

Мое дерево каталогов

enter image description here и код файла c.

#include <jni.h>
#include <stdio.h>
#include <stdlib.h>
#include <cpu-features.h>

jstring
Java_androidFfmpeg_ArmArchHelper_cpuArchFromJNI(JNIEnv* env, jobject       obj)
{
char arch_info[11] = "";

if (android_getCpuFamily() == ANDROID_CPU_FAMILY_ARM) {
    strcpy(arch_info, "ARM");

    uint64_t cpuFeatures = android_getCpuFeatures();
    if ((cpuFeatures & ANDROID_CPU_ARM_FEATURE_ARMv7) != 0) {
        strcat(arch_info, " v7");

        if((cpuFeatures & ANDROID_CPU_ARM_FEATURE_NEON) != 0) {
            strcat(arch_info, "-neon");
        }
    }
}
return (*env)->NewStringUTF(env, arch_info);
}

А это оценка (приложение)

android {
compileSdkVersion 27
buildToolsVersion '27.0.3'
defaultConfig {
    applicationId "com.softcode.kihnoplay"
    minSdkVersion 21
    targetSdkVersion 25
    versionCode 1
    versionName "1.00"
    testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"

    renderscriptTargetApi 19
    renderscriptSupportModeEnabled true

    ndk {
        abiFilters "armeabi", "armeabi-v7a","x86"
    }

    aaptOptions
            {
                cruncherEnabled = false
            }
}

buildTypes {
    debug {
        minifyEnabled false
        proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
    }
    release {
        minifyEnabled false
        proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
    }
}
sourceSets.main {
    jniLibs.srcDir 'src/main/libs'
    jni.srcDirs = [] //important!!!! disable automatic ndk-build call
}

Я не знаю, почему происходит эта ошибка. Если вы знаете эту проблему, пожалуйста, дайте мне знать. Спасибо тебе!

1 Ответ

0 голосов
/ 02 июля 2018

Полагаю, вы используете clang ++ для компиляции исходного кода c, верно? Если это правда, попробуйте изменить имя файла на armArch.cpp и добавить ключевое слово extern в заголовок C и исходные файлы. например

#ifdef __cplusplus
extern "C" {
#endif

// your function declarations or implementations. 

#ifdef __cplusplus
}
#endif

Кроме того, я хотел бы предложить использовать externalNativeBuild и cmake вместо более старого ndk блока, который не является современным соглашением для собственных сборок Android. Например.

android {

    //....

    defaultConfig {
        applicationId "com.softcode.kihnoplay"
        minSdkVersion 21
        targetSdkVersion 25
        versionCode 1
        versionName "1.00"
        testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"

        externalNativeBuild {
            cmake {
                cppFlags "-std=c++11 -frtti -fexceptions"
                abiFilters 'armeabi-v7a', 'arm64-v8a', 'x86', 'x86_64' 
            }
        }
    }
    buildTypes {
        release {
            //....
        }
    }

    externalNativeBuild {
        cmake {
            path "CMakeLists.txt"
        }
    }
}

Android NDK устарел armeabi , вам на самом деле не нужно включать этот ABI в настройках abiFilters. ABI 'armeabi-v7a', 'arm64-v8a', 'x86', 'x86_64' будет достаточно для поддержки практически всех существующих устройств Android, а также эмуляторов.

Подробное руководство по JNI можно найти в https://arophix.com/2017/12/17/andoid-jni-summary/ и пример проекта JNI от https://github.com/russell-shizhen/JniExample

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...