jni函数除了要靠c++代码实现功能之外,在一些情况下还需要调用java里的方法来实现一些功能。
解决问题:在jni函数中调用特定java类的特定方法。
新建一个java类:
package com.example.hello_cmake;
import android.util.Log;
public class TestCallBack {
static {
System.loadLibrary("native-lib");
}
//回调方法 里面调用了add
public native void callBackAdd();
public int add(int x,int y){
Log.d("CALLBACK","add x= "+x+"y="+y);
return x+y;
}
}
在TestCallBack类中声明了两个函数,一个是add,这个函数显然是java方法;另一个是CallBackAdd,该方法是个jni函数。
我们在该jni函数实现调用add的功能。
extern "C"
JNIEXPORT void JNICALL
Java_com_example_hello_1cmake_TestCallBack_callBackAdd(JNIEnv *env, jobject jobj) {
//1得到字节码
jclass jclazz = env->FindClass("com/example/hello_cmake/TestCallBack");
//2得到方法
jmethodID jmethodIds = env->GetMethodID(jclazz,"add","(II)I");
//3实例化
jobject object = env->AllocObject(jclazz);
//4调用方法
env->CallIntMethod(object,jmethodIds,100,1);
}
从代码中可以看出,当jni函数需要调用java方面的东西的时候,env和jobj就派上用场了。
第一步:得到字节码,也就是得到要调用的java方法所属的类。
这里需要注意com.example.hello_cmake.TestCallBack是类的完整路径,但是识别的时候是以Linux标准进行的,因此里面的.要替换成/来表示路径。
findclass方法是泛用性比较强的得到字节码的方式,通过该函数可以令jni函数任意调用java各个类里面的java方法。另外一种得到字节码的方式是:
jclass a_class = env->GetObjectClass(instance);
getobject方法只能得到jni函数所属的类的字节码,也就是说只能调用与该jni函数所属同类的java方法,泛用性大打折扣。
第二步:得到要调用的java方法的id。
这里注意参数中第二个是java方法的名字,第三个是java方法的签名。
第三步:实例化一个对象。
在java中,类的成员方法必须以实例调用的形式调用,因此必须实例化一个对象,才能调用成员函数add。
第四步:调用方法。
调用方法的函数参数中,前两个分别为实例与方法的ID,后面的参数则是调用的java方法的参数。
jni函数本身也算是java类的一个成员方法,在调用的时候,需要先实例化TestCallBack类,再进行调用。