使本机代码访问 java 方法和数据成员

make native code access java methods and data members

我正在使用本机代码开发一个 android 项目,该项目应该更新对象中的列表和其他一些布尔变量。

考虑以下代码

一些 java class 在我的代码中看起来像:

 class ReturnObject
{
    boolean a, b;
    public List<String[]> listA;

}

public class foo
{
    public native void someFunction(ReturnObject returnObject);
}

本机代码如下所示:

JNIEXPORT void JNICALL Java_com_example_androidtest_TestActivity_someFunction
  (JNIEnv * env, jobject jObj, jobject returnObject) {
    std::string f = "foo";
    // missing code here
    // returnObject.add(f) // add like in java
}

如何设置布尔变量的值?

对于列表(知道我的数据在本机代码中存储为 std::string),我如何调用 add(String[] string)来自本机代码的列表 class 的方法?

这里是设置布尔值的示例代码,按照相同的准则调用列表的add方法。

#include <stdio.h>
#include <jni.h>
#include "com_example_Foo.h"

JNIEXPORT void JNICALL Java_com_example_Foo_someFunction
(JNIEnv *env, jobject object, jobject returnObject)
{
    jclass clzReturnObject = env->FindClass("com/example/ReturnObject");
    jfieldID fieldA = env->GetFieldID(clzReturnObject, "a", "Z");
    jfieldID fieldB = env->GetFieldID(clzReturnObject, "b", "Z");

    env->SetBooleanField(returnObject, fieldA, true);
    if (env->ExceptionCheck()) {
        fprintf(stderr, "error set boolean for a");
        env->ExceptionDescribe();
    }
    env->SetBooleanField(returnObject, fieldB, false);
    if (env->ExceptionCheck()) {
        fprintf(stderr, "error set boolean for b");
        env->ExceptionDescribe();
    }

    jfieldID fieldListID = env->GetFieldID(clzReturnObject, "list",
            "Ljava/util/List;");
    jobject listObject = env->GetObjectField(returnObject, fieldListID);

    jclass clzList = env->FindClass("java/util/List");
    jmethodID addMethodID = env->GetMethodID(clzList, "add",
            "(Ljava/lang/Object;)Z");

    jclass clzString = env->FindClass("java/lang/String");

    jstring initElement = env->NewStringUTF("0000");

    jobjectArray toAdd = env->NewObjectArray(10, clzString, initElement);

    if (env->ExceptionCheck()) {
        fprintf(stderr, "error create string array");
        env->ExceptionDescribe();
    }

    env->CallBooleanMethod(listObject, addMethodID, toAdd);
    env->DeleteLocalRef(initElement);
}

以下是java代码

public class Foo {

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

public static void main(String[] args) {

    ReturnObject returnObject = new ReturnObject();
    returnObject.list = new ArrayList<>();

    Foo foo = new Foo();
    foo.someFunction(returnObject);
    System.out.println("size is " + returnObject.list.size());

    foo.someFunction(returnObject);
    System.out.println("size is " + returnObject.list.size());

    foo.someFunction(returnObject);
    System.out.println("size is " + returnObject.list.size());

    // we have three element added from JNI
    for (String string : returnObject.list.get(2)) {
        System.out.println(string);
    }
}

public native void someFunction(ReturnObject returnObject);
}

这里是调用字符串构造函数在JNI中创建jstring的例子。

jbyteArray initValue = env->NewByteArray(10);
env->SetByteArrayRegion(initValue, 0, 9, (const signed char*)"ABCDEFG");
jstring enc = env->NewStringUTF("UTF-8");
jclass clzString = env->FindClass("java/lang/String");
jmethodID cMethodID = env->GetMethodID(clzString, "<init>", "([BLjava/lang/String;)V");
jstring result = reinterpret_cast<jstring>(env->NewObject(clzString, cMethodID, initValue, enc));

以下是从 JNI 读取 EditText 文本的示例。

void Java_com_example_foo_MainActivity_someFunction(JNIEnv* env, jobject object,
    jstring objString, jobject objEditText) {
    __android_log_print(ANDROID_LOG_INFO, "Foo", "%s\n", "JNI call");

    // get edittext value
    jclass clzEditText = env->GetObjectClass(objEditText);
    jmethodID methodGetText = env->GetMethodID(clzEditText, "getText", "()Landroid/text/Editable;");

    jobject objEditable = env->CallObjectMethod(objEditText, methodGetText);
    jclass clzEditable = env->GetObjectClass(objEditable);
    jmethodID methodToString = env->GetMethodID(clzEditable, "toString", "()Ljava/lang/String;");

    jstring editText = reinterpret_cast<jstring>(env->CallObjectMethod(
        objEditable, methodToString));

    jboolean isCopy = true;
    __android_log_print(ANDROID_LOG_INFO, "Foo", "From java: %s\n", env->GetStringUTFChars(editText, &isCopy));
}

假设在您的应用 MainActivity 中声明了一个 jni。

private native void someFunction(String s, EditText editText);

我没有找到直接从 JNI 设置 String 的好的解决方案。如果你想在 JNI 中操作 String,你可以将结果从 return 值传递给 Java。

private native String Foo();