JNI访问属性和方法

导读

《JNI之动态注册与静态注册》 一文中,我们介绍了JNI函数动态注册的同时也介绍了JNI中的属性描述符和函数描述符。对这两个概念还不熟悉的童鞋们需要再温习一下。
在今天的文章中我们主要介绍在JNI中如何访问java层对象的属性以及调用java层对象的成员方法等相关知识。

访问java成员属性

1、访问普通属性
在JNI访问java类属性分为两个步骤,首先是通过FindClass函数找到对应的类,然后通过GetFieldID找到对应的属性,如果需要修改变量的话则通过一系列的SetTypeField函数进行修改即可.

下面的例子通过JNI的方式访问了java类Person的name属性并作出修改:

Person.java

public class Person {
    private static int defaultAge = 18;
    private int age;
    private String name;
    Person(int age,String name){
        this.age = age;
        this.name = name;
    }
    public void printName(){
        Log.v("PersonTag","my name is:" + name);
    }

    public static void printDefaultAge(){
        Log.v("PersonTag","default age is:" + defaultAge);
    }

    // 访问名字属性
    public native void changeName();
    // 访问静态属性
    public native void accessStatic();
    // 通过JNI调用java的方法
    public native void printNameByJNI();
    // 访问静态方法
    public native static void callStaticFunc();
    // JNI调用构造方法,生成对象
    public native static Person createPersonByJNI();
}
MainActivity.java

public class MainActivity extends AppCompatActivity {

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

    private ActivityMainBinding binding;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        binding = ActivityMainBinding.inflate(getLayoutInflater());
        Person person = new Person(20,"jack");
        person.changeName();
        person.printName();
    }

}
native-lib.cpp

extern "C"
JNIEXPORT void JNICALL
Java_com_fly_jnitest_Person_changeName(JNIEnv *env, jobject thiz) {
    // 获取对应的类
    jclass jclazz = env->FindClass("com/fly/jnitest/Person");
    // 获取属性,第三个参数是属性描述符
    jfieldID nameFielid = env->GetFieldID(jclazz,"name","Ljava/lang/String;");
    // 修改成员属性
    env->SetObjectField(thiz,nameFielid,env->NewStringUTF("james"));
}

2、访问静态属性
访问静态属性和访问普通属性的流程是一样的,只不过获取的静态属性的函数变成了GetStaticFieldID,还是以上面的代码作为例子,我们通过JNI的方式访问Person类的静态属性defaultAge并作出修改:

extern "C"
JNIEXPORT void JNICALL
Java_com_fly_jnitest_Person_accessStatic(JNIEnv *env, jobject thiz) {
    // 获取对应的类
    jclass jclazz = env->FindClass("com/fly/jnitest/Person");
    // 获取静态属性,第三个参数是属性描述符
    jfieldID defaultAgeFielid = env->GetStaticFieldID(jclazz,"defaultAge", "I");
    // 修改静态属性
    env->SetStaticIntField(jclazz,defaultAgeFielid,19);
}

调用java成员方法

1、调用普通方法
同访问类的成员属性一样,在JNI中调用java层类的成员方法时也需要先找到对应的类,然后通过JNI函数GetMethodID获取到对应的方法id,就可以使用函数CallVoidMethodCallObjectMethod等完成调用了。

继续是以上面的代码为例,通过JNI的方式,调用Person类对象的printName方法:

extern "C"
JNIEXPORT void JNICALL
Java_com_fly_jnitest_Person_printNameByJNI(JNIEnv *env, jobject thiz) {
    // 获取对应的类
    jclass jclazz = env->FindClass("com/fly/jnitest/Person");
    // 获取方法id,第三个参数是函数签名
    jmethodID methodId = env->GetMethodID(jclazz,"printName", "()V");
    // 调用java方法
    env->CallVoidMethod(thiz,methodId);
}

2、调用静态方法
调用静态方法与调用普通方法步骤一致,只不过获取方法id的函数换成了GetStaticMethodID,调用的函数换成CallStaticVoidMethod即可,这里就不贴代码了。

3、调用构造函数

调用构造函数和调用普通成员方法一样,也是先获取到对应的类,然后获取到构造函数的jmethodID,最后通过NewObject即可生成java对象,但在获取构造函数的jmethodID时,构造函数的方法名固定为
以下例子展示了在JNI层创建一个Person类并返回给java层:

extern "C"
JNIEXPORT jobject JNICALL
Java_com_fly_jnitest_Person_createPersonByJNI(JNIEnv *env, jclass clazz) {
    // 获取对应的类
    jclass personClazz = env->FindClass("com/fly/jnitest/Person");
    // 获取构造方法,构造方法的函数名称固定为
    jmethodID methodId = env->GetMethodID(personClazz,"", "(ILjava/lang/String;)V");
    // 通过构造方法生成对象
    jobject person = env->NewObject(personClazz,methodId,20,env->NewStringUTF("james"));
    return person;
}

系列推荐

JNI基础简介
JNI之数组与字符串的使用
JNI之动态注册与静态注册

关注我,一起进步,人生不止coding!!!

你可能感兴趣的:(androidjnindk)