导读

在《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,就可以使用函数​​CallVoidMethod​​、​​CallObjectMethod​​等完成调用了。

继续是以上面的代码为例,通过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时,构造函数的方法名固定为​​<init>​​。

以下例子展示了在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");
// 获取构造方法,构造方法的函数名称固定为<init>
jmethodID methodId = env->GetMethodID(personClazz,"<init>", "(ILjava/lang/String;)V");
// 通过构造方法生成对象
jobject person = env->NewObject(personClazz,methodId,20,env->NewStringUTF("james"));
return person;
}

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