动态注册的核心思想是找到对应的class,然后在JavaEnv指针中调用RegisterNatives(clazz, getMethods, numMethods)函数进行注册。
注
静态注册的缺点
- 需要编译所有声明了native函数的Java类,为每个生成的class文件用javah生成一个头文件
- javah生成的头文件函数名特别长
- 初次调用native函数时需要根据函数名字搜索对应的JNI函数来建立关联关系,会影响运行效率。
1.编写被注册的函数
cpp
extern "C" JNIEXPORT jstring JNICALL
processFile(
JNIEnv *env,
jobject /* this */) {
std::string hello = "Hello from processFile";
return env->NewStringUTF(hello.c_str());
}
java
public native String processFile();
2.注册函数数组
static JNINativeMethod getMethods[] = {
{
"processFile", //函数名
"()Ljava/lang/String;", //参数与返回值
(void *) processFile //函数指针,一定要强转为void*
}
};
3.注册
jclass clazz;
clazz = env->FindClass(className);
env->RegisterNatives(clazz, getMethods, numMethods);
4.注册的位置
因为JNI初始化时会调用JNI_Onload函数,所以复写JNI_Onload函数,并把JNI函数注册放在这里。
jint JNI_OnLoad(JavaVM* vm, void* reserved)
{
JNIEnv* env = NULL;
jint result = -1;
if(vm->GetEnv((void**) &env, JNI_VERSION_1_6 != JNI_OK)) {
LOGW(1, "get env failed")
return JNI_ERR;
}
//放在这里
jclass clazz;
clazz = env->FindClass(className);
env->RegisterNatives(clazz, getMethods, numMethods);
return JNI_VERSION_1_6;
}
5.例子
int register_jni_method(JNIEnv *env, const char* className, const JNINativeMethod* getMethods, int numMethods){
jclass clazz;
clazz = env->FindClass(className);
if(env->RegisterNatives(clazz, getMethods, numMethods) < 0)
return -1;
return 0;
}
extern "C" JNIEXPORT jstring JNICALL
processFile(
JNIEnv *env,
jobject /* this */) {
std::string hello = "Hello from processFile";
return env->NewStringUTF(hello.c_str());
}
static JNINativeMethod getMethods[] = {
{
"processFile",
"()Ljava/lang/String;",
(void *) processFile
}
};
jint JNI_OnLoad(JavaVM* vm, void* reserved)
{
JNIEnv* env = NULL;
jint result = -1;
if(vm->GetEnv((void**) &env, JNI_VERSION_1_6 != JNI_OK)) {
// LOGW(1, "get env failed")
return JNI_ERR;
}
if (register_jni_method(env, "com/example/max/opencvtest/MainActivity",getMethods,1) < 0){
// LOGW(1, "register failed")
return JNI_ERR;
}
return JNI_VERSION_1_6;
}