导读
在《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;
}