前言
在android上进行ndk开发时,会无法避免的java层调用c层代码,那么,有时候就需要c层调用java层的代码了,我们如何进行c层调用java层的方法呢,这就是这篇简书需要解决的问题了。
正文
我们新建一个包含c++支持的项目过后会自动出现native-lib.cpp文件,这是初始就新建好了的,那么我们就在这个文件中操作。
jni访问java层非静态方法。
先创建一个native方法。
public native void accessMethod();
创建一个java方法,以便在c层调用,用最简单的showtoast为例子。
void showToast(){
Toast.makeText(this,"hello jni",Toast.LENGTH_LONG).show();
}
然后在native-lib.cpp中,
JNIEXPORT void JNICALL
Java_com_ndk_1demo_MainActivity_accessMethod(JNIEnv *env, jobject instance) {
// 得到jclass
jclass jclz = env->GetObjectClass(instance);
//得到方法名
//GetMethodID参数说明:jclass,调用的对应的java层的方法名,方法签名。
jmethodID mid = env->GetMethodID(jclz,"showToast","()V");
env->CallVoidMethod(instance,mid);
}
关于方法签名,有两个方法来实现:
1.使用javap -s -p xxxx.class
a.找到方法对用类的.class文件,并cd到这个文件目录中,
b.使用命令即可以查看各个方法的签名。
2.自己依照规则编写,
jni访问java层静态方法。
GetMethodID--->GetStaticMethodID
CallVoidMethod--->CallStaticVoidMethod
先创建一个native方法。
public native int accessStaticMethod();
创建一个java静态方法,以便在c层调用,两个整数比较取较大值。
static int maxNum(int a,int b){
return a>b?a:b;
}
然后在native-lib.cpp中,
extern "C"
JNIEXPORT jint JNICALL
Java_com_ndk_1demo_MainActivity_accessStaticMethod(JNIEnv *env, jobject instance) {
//得到jclass
jclass jclz = env->GetObjectClass(instance);
//得到静态方法的方法名
jmethodID jmid = env->GetStaticMethodID(jclz,"maxNum","(II)I");
//调用方法
int max = env->CallStaticIntMethod(jclz,jmid,5,4);
LOGE("LC %d",max);
return max;
}
jni访问java构造方法。
这里以调用java自带的Date方法为例。
先创建一个native方法。
public native Date accessDateMethod();
然后在native-lib.cpp中,
extern "C"
JNIEXPORT jobject JNICALL
Java_com_ndk_1demo_MainActivity_accessDateMethod(JNIEnv *env, jobject instance) {
//通过类的路径从jvm里面找到对应的类
//JNIEnv类中有如下几个简单的函数可以取得jclass
//jclass FindClass(const char* clsName) 根据类名来查找一个类,完整类名。
//jclass GetObjectClass(jobject obj) 根据一个对象,获取该对象的类
//jclass GetSuperClass(jclass obj) 获取一个类的父类
jclass jclz = env->FindClass("java/util/Date");
//因为是构造函数,所以name都是<init>
jmethodID jmid=env->GetMethodID(jclz,"<init>","()V");
//实例话一个date对象
jobject jobj = env->NewObject(jclz,jmid);
//得到date对象中的方法,此处调用的是getTime方法,可在Date这个类里面查找到
jmethodID tmid = env->GetMethodID(jclz,"getTime","()J");
//调用其中方法
jlong time = env->CallLongMethod(jobj,tmid);
LOGE("LCssssss time:%lld\n",time);
return jobj;
}
其中注意事项都在代码中有详细备注,自己仔细阅读。
总结
jni调用java层方法差不多就这么三大类,但是步奏差不多分为三步:
1.得到jclass
2.得到jmethodId
3.调用jmethodId
源码地址