无法解析相应的 JNI 函数 Java_com_mozilla_greetings_RustGreetings_greeting

Cannot resolve corresponding JNI function Java_com_mozilla_greetings_RustGreetings_greeting

我正在尝试复制 this tutorial for building and using Rust lib in Android app, I build the library successful, and uploaded the generated libs here

Android使用Java_<Package>_Class_function主题需要调用的函数是:

Java_com_mozilla_greetings_RustGreetings_greeting

我的 android 应用结构如下:

我的 JNI 包装器出现以下错误:

Cannot resolve corresponding JNI function Java_com_mozilla_greetings_RustGreetings_greeting

JNI 包装器是:

package com.mozilla.greetings;

public class RustGreetings {

    private static native String greeting(final String pattern);

    public String sayHello(String to) {
        return greeting(to);
    }
}

而主要的 class 是:

package com.mozilla.greetings;

import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.widget.TextView;

public class GreetingsActivity extends AppCompatActivity {

    static {
        System.loadLibrary("greetings");
    }

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

        RustGreetings g = new RustGreetings();
        String r = g.sayHello("world");
        ((TextView) findViewById(R.id.greetingField)).setText(r);
    }
}

我使用 JNA 解决了它,我认为它比 JNI 慢,我将在下面使用 JNA 写下我的解决方案,希望有人使用 JNI 提供所需的修复]

我的应用程序结构如下,使用 kotlin:

  • 我将libjnidispatch.so添加到每个库构建文件夹,,这可以通过从here中提取所需的android架构来获得,下载所需的jar,然后提取它以获得libjnidispatch.so
  • 我为 jna
  • 创建了界面

JNA.kt

package com.mozilla.greetings

import com.sun.jna.Library

   interface JNA : Library {
      fun rust_greeting(pattern: String): String
   }
  1. 我为 jna 创建了包装器,RustGreetings.kt:

    package com.mozilla.greetings
    
    import com.sun.jna.Native
    
    class RustGreetings {
    
    fun sayHello(to: String): String =
            Native.loadLibrary<JNA>("greetings", JNA::class.java).rust_greeting(to)
    }
    
  2. 主要activity,GreetingsActivity.kt:

    package com.mozilla.greetings
    
    import android.support.v7.app.AppCompatActivity
    import android.os.Bundle
    import kotlinx.android.synthetic.main.activity_greetings.*
    
    
    class GreetingsActivity : AppCompatActivity() {
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_greetings)
    
        val g = RustGreetings()
        val r = g.sayHello("Rust")
        greetingField.text = r
    }
    
    companion object {
        init {
            System.loadLibrary("greetings")
        }
    }
    }
    

注意: 为了避免使用 findViewById,我使用了 kotlin 扩展,如 here 所述,并将以下内容添加到 build.gradle(模块):

apply plugin: 'kotlin-android-extensions'

更新

看起来这是一个 IDE 问题,代码没有任何问题,应用程序已被执行。 我用 Kotlin 重写了它,它也执行得很顺利,在我的 kotlin 代码下面:

GreetingsActivity.kt

package com.mozilla.greetings

import android.support.v7.app.AppCompatActivity
import android.os.Bundle
import kotlinx.android.synthetic.main.activity_greetings.*

class GreetingsActivity : AppCompatActivity() {

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_greetings)

        val g = RustGreetings()
        val r = g.sayHello("My Rust")
        greetingField.text = r
    }

    companion object {

        init {
            System.loadLibrary("greetings")
        }
    }
}

RustGreetings.kt

package com.mozilla.greetings

class RustGreetings {

    private external fun greeting(pattern: String): String

    fun sayHello(to: String): String = greeting(to)
}

build.gradle

apply plugin: 'com.android.application'
apply plugin: 'kotlin-android-extensions'

apply plugin: 'kotlin-android'

apply plugin: 'kotlin-android-extensions'

android {
    compileSdkVersion 28
    defaultConfig {
        applicationId "com.mozilla.greetings"
        minSdkVersion 28
        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"org.jetbrains.kotlin:kotlin-stdlib-jdk7:$kotlin_version"
    implementation 'com.android.support:appcompat-v7:28.0.0'
    implementation 'com.android.support.constraint:constraint-layout:1.1.3'
    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'
}

activity_greetings.xml

<?xml version="1.0" encoding="utf-8"?>
<android.support.constraint.ConstraintLayout
        xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:tools="http://schemas.android.com/tools"
        xmlns:app="http://schemas.android.com/apk/res-auto"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        tools:context=".GreetingsActivity">

    <TextView
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="Hello World!"
            app:layout_constraintBottom_toBottomOf="parent"
            app:layout_constraintLeft_toLeftOf="parent"
            app:layout_constraintRight_toRightOf="parent"
            app:layout_constraintTop_toTopOf="parent" android:id="@+id/greetingField"/>

</android.support.constraint.ConstraintLayout>

在结构和执行下面,apk 是 here: