Android JNI的初步使用--2 动态加载

Android JNI的初步使用--1
上一篇做了一个jni静态加载的示例,这一篇来看一下jni的方法的动态加载。
我先展示一下cpp中动态加载的示范代码:

jstring fun1(JNIEnv *env, jobject jobj) {
    std::string hello = "动态注册";
    return env->NewStringUTF(hello.c_str());
}

jstring fun2(JNIEnv *env, jobject jobj) {
    std::string hello = "动态注册";
    LOGD("%s", hello.c_str());
    return env->NewStringUTF(hello.c_str());
}
static const char *mClassName = "com/hao/minovel/jni/DragCrash";

static const JNINativeMethod mMethods[] = {
        //参数一:java方法名
        //参数二:()中为调用时需要的传残参类型后面为返回类型
        //参数三:jni中对应的方法
        {"getString2", "()Ljava/lang/String;",                           (jstring *) fun1},
        {"initDate",   "(Lcom/hao/minovel/jni/Test;)Ljava/lang/String;", (jstring *) fun2},
};

//在调用java的load的时候 会调用此方法
JNIEXPORT jint JNICALL JNI_OnLoad(JavaVM *vm, void *res) {
   JNIEnv *env = NULL; //获得 JniEnv
   int r = vm->GetEnv((void **) &env, JNI_VERSION_1_4);
   if (r != JNI_OK) { return -1; }
   jclass mainActivityCls = env->FindClass(mClassName); // 注册 如果小于0则注册失败
   r = env->RegisterNatives(mainActivityCls, mMethods, 2);//第三个参数为你需要加载的mMethods中的数量
   if (r != JNI_OK) { return -1; }
   return JNI_VERSION_1_4;
}

在上面的代码中,我们需要关注的是JNI_Onload中的两个方法

   jclass mainActivityCls = env->FindClass(mClassName); // 注册 如果小于0则注册失败
   r = env->RegisterNatives(mainActivityCls, mMethods, 2);//第三个参数为你需要加载的mMethods中的数量

其中mClassName为java中的native方法的类路径,FindClass方法会通过此路径获取到需要加载的类ID;
而RegisterNatives则通过获取的类ID和方法介绍及方法加载数来注册native方法。
ps:注册多个方法时,需要记得修改RegisterNatives方法的第3个参数。

而RegisterNatives方法中的第二个参数mMethods中包含的是需要注册的ative方法重要信息,mMethods是一个jni.h中的一个结构体,代码如下:

typedef struct {
    const char* name;//方法名
    const char* signature;//方法签名
    void*       fnPtr;//jni中的方法名和返回类型
} JNINativeMethod;

name:java中native方法的方法名。
signature:java方法的签名。
例如:(I)Ljava/lang/String;表示该方法的是传入了一个int类型的参数,返回一个String。具体类型解释见下面的签名介绍;
fnPtr:jni中对应的方法。
例如:(jstring *) fun2 对应的就是 cpp中的fun2方法,()为方法的返回类型。

java中的签名介绍:

Java类型  相应的签名                                       例子
boolean     Z
byte        B
char        C
short       S
int         I
long        L
float       F
double      D
void        V
Object      L用"/"分割的完整类名;                      ex: Ljava/lang/String;
Array       [签名                                      ex: [I   [Ljava/lang/String;
Method      (参数1类型签名参数2类型签名...)返回值类型签名

我们也可以通过javap命令来获取方法的签名;
命令:javap -s -p 类.class

  public static native java.lang.String getString2();
    descriptor: ()Ljava/lang/String;

  public static native java.lang.String initDate(com.hao.minovel.jni.Test);
    descriptor: (Lcom/hao/minovel/jni/Test;)Ljava/lang/String;

这个是我获取的测试两个方法的签名。
以上是我对jni动态注册的理解,欢迎大佬指正。
Android JNI的初步使用--3

你可能感兴趣的:(Android JNI的初步使用--2 动态加载)