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