RegisterNatives的使用方法
方式一
在Android中通过jni调用本地方法(c/c++),通常使用javah生成规范的头文件,定义一个函数实现本地方法,函数名必须使用本地方法的全类名的规范来写。
Java_ + 包名 + 类名+ 接口名
注意名字之间用下划线,下面是示例:
JNIEXPORT jstring Java_com_example_test_MainActivity_helloworld(JNIEnv *, jclass );
有没有觉得这种写法太反人类了(感谢很麻烦有没有),其实jni还提供了RegisterNative函数手动的注册native方法,该方法可以自由命名函数,不必像上述方法那样拘泥特定烦杂的命名方式。
方式二
RegisterNatives使用示例:
static int registerNativeMethods(JNIEnv* env)
{
jclass clazz;
clazz = env->FindClass("com/example/test/MainActivity");
if (clazz == NULL) {
return JNI_FALSE;
}
if (env->RegisterNatives(clazz, gMethods, sizeof(gMethods) / sizeof(gMethods[0])) < 0) {
return JNI_FALSE;
}
return JNI_TRUE;
}
RegisterNatives中第二个参数gMethods是一个二维数组,代表着这个class里的每一个native方法所对应的实现的方法。写法如下示例:
static JNINativeMethod gMethods[] = {
{"helloworld", "()Ljava/lang/String;", (void*)Jni_helloworld},
{"fun2_java", "()Ljava/lang/String;", (void*)fun2_jni},
};
第三个参数代表要指定的native的数量。
这时将前面在jni中声明的
jstring JNIEXPORT jstring Java_com_example_test_MainActivity_helloworld(JNIEnv *, jclass );
改为
jstring helloworld(JNIEnv *, jclass);
是不是清爽了很多!
JNI字段描述符
在使用RegisterNatives注册本地方法中比较容易出错的地方就是JNINativeMethod gMethods[]其中的()Ljava/lang/String;。
先解释下**()Ljava/lang/String;**的含义,它是一种对函数返回值和参数的编码,这种编码叫做JNI字段描述符(JavaNative Interface FieldDescriptors)。
JNI字段描述符的规则是在括号里放置参数,在括号后面放置返回类型:
(参数描述符)返回类型 // JNI 字段描述符
例如**()Ljava/lang/String;表示对应的java中的helloworld方法无传入参数,返回类型为String,即String helloworld();
这个示例可能过于简单,下面再举几个例子:
(Ljava/lang/String;Ljava/lang/Object;)Ljava/lang/String;表示String f(String a,Object b);
([B)Ljava/lang/String;表示String f(byte [] bytes);这里B表示java中的byte类型,[表示数组**,[B表示byte []。
以此类推,
[Ljava/lang/String;表示String []
[Ljava/lang/Object;表示Object[]
如果是二维数组int[][],用[[I表示。
另外要注意的是,引用类型(除基本类型的数组外)通常以"L"开头,以";“结尾,中间是用”/" 隔开的包及类名。所以";"分号是属于引用类型的一部分,因此参数中如果有多个参数的话是不用;间隔的。如:
(BI)V 表示 void f(byte b, int i);
(ILjava/lang/String;)V 表示 void f(int i,String s);
最后附上java类型与jni中字符对应的关系表: