Android - NDK: 在jni层生成java层对象,并调用java层的方法
Android - NDK
一、在JNI层生成java层的对象,并调用java层的方法
1、java类的定义
import android.util.Log;
public class MyCustomObj {
private final static String TAG = MyCustomObj.class.getSimpleName();
private String name;
private int age;
// public MyCustomObj(String name,int age){
// this.name = name;
// this.age = age;
// }
public String introduce(){
Log.d(TAG, "introduce: name = " + name + ", age = " + age);
return this.name + " , " + this.age;
}
@Override
public String toString() {
return "MyCustomObj{" +
"name='" + name + '\'' +
", age=" + age +
'}';
}
}
2、在MainActivity中声明native方法并调用
/**
* 在Jni中生成java对象,并调用java代码
* 在c中生成java对象的属性值
* @return
*/
public native String cCreateJobj();
// 在jni层生成java对象并调用java对象的方法
binding.btnCreateJObj.setOnClickListener(view->{
String strRet = cCreateJobj();
Toast.makeText(this,strRet,Toast.LENGTH_SHORT).show();
});
3、在cpp文件中实现native方法
extern "C"
JNIEXPORT jstring JNICALL
Java_com_hisign_ndktest_MainActivity_cCreateJobj(JNIEnv *env, jobject thiz) {
// 1 先找到java类,java类的包名+类名,包名中间用“/”分割
jclass myCustomObjClass = env->FindClass("com/hisign/ndktest/MyCustomObj");
// 2 获取类的构造函数
// 有参构造函数(固定的写法"<init>")
// jmethodID constructor = env->GetMethodID(myCustomObjClass,"<init>","(Ljava/lang/String;I)V");
// 无参构造函数
jmethodID voidConstructor = env->GetMethodID(myCustomObjClass, "<init>", "()V");
// 获取对象的方法,返回值是String类型
jmethodID introduce = env->GetMethodID(myCustomObjClass, "introduce", "()Ljava/lang/String;");
// 3 创建java对象(调用有参的构造函数,并传递参数)
//jobject myCustomOjb = env->NewObject(myCustomObjClass,constructor,jNameValue,jAgeValue);
// 创建java对象(调用无参的构造函数)
jobject myCustomOjb = env->NewObject(myCustomObjClass, voidConstructor);
// 4 在c层中创建变量的值
const char *name = "lixm";
const int age = 20;
// 把c层的值转换为jni层的值
jstring jNameValue = env->NewStringUTF(name);
jint jAgeValue = age;
//5 创建对象的属性
jfieldID jName = env->GetFieldID(myCustomObjClass,"name", "Ljava/lang/String;");
jfieldID jAge = env->GetFieldID(myCustomObjClass,"age", "I");
// 6 给对象的属性赋值
env->SetObjectField(myCustomOjb,jName,jNameValue);
env->SetIntField(myCustomOjb,jAge,jAgeValue);
// 7 调用java对象的方法
//env->CallIntMethod(myCustomOjb,introduce); // 返回值是int类型方法调用
jstring obj = (jstring) env->CallObjectMethod(myCustomOjb, introduce); //返回值jobject -->jstring
return obj; // 返回jstring
}
二、总结
1、FindClass
传入的参数为类的全路径,包名中间用“/”分割,通过反射的机制找到这个类
jclass myCustomObjClass = env->FindClass("com/hisign/ndktest/MyCustomObj");
接下来获取method和java类对象都需要基于jclass.
2、获取对象的方法 GetMethodID
// 获取类的构造方法
jmethodID voidConstructor = env->GetMethodID(myCustomObjClass, "<init>", "()V");
// 获取类的普通方法
jmethodID introduce = env->GetMethodID(myCustomObjClass, "introduce", "()Ljava/lang/String;");
获取的方法为构造函数时,GetMethodID的第二个参数(方法名称)固定写为" ",
普通方法时,GetMethodID的第二个参数为实际对应的方法的名称。
3、生成java对象 NewObject
基于之前生成的jclass和构造方法,创建java对象
jobject myCustomOjb = env->NewObject(myCustomObjClass, voidConstcructor);
4、调用java对象的方法
env->CallObjectMethod(myCustomOjb, introduce)
CallObjectMethod的第一个参数是java对象,第二个参数方法名称,如果方法需要参数时,在第三个,第四个…依次传入。