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

时间:2015-03-02 14:43:21

标签: java android c++ android-ndk java-native-interface

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

考虑以下代码

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

 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
}

如何设置布尔变量的值?

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

3 个答案:

答案 0 :(得分:2)

以下是设置布尔值的示例代码,请遵循相同的指南来调用列表的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);
}

答案 1 :(得分:0)

这是一个调用字符串构造函数在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));

答案 2 :(得分:0)

以下是从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,可以从返回值将结果传递给Java。

private native String Foo();