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函数签名。
第三步:实例化一个对象。
在java中,类的成员方法必须以实例调用的形式调用,因此必须实例化一个对象,才能调用成员函数add。
第四步:调用方法。
调用方法的函数参数中,前两个分别为实例与方法的ID,后面的参数则是调用的java方法的参数。
jni函数本身也算是java类的一个成员方法,在调用的时候,需要先实例化TestCallBack类,再进行调用。