概述
函数注册分为静态注册和动态注册。
本文从理论分析和具体操作案例两个方面讨论。
静态注册
简介
以java“包名+类名+函数名”命名jni中的函数,java调用时自动去jni export 的函数中寻找,并执行其中的命令。函数中的指令多数为java->c变量转换,然后调用so中的主要功能函数。
优
- 自动化生成 – 可以使用javah这一指令自动化生成这一系列函数,大大减少人工手写;
- 明了易读 – 由于jni函数名和java中对应,对应关系简单明了,非开发人员使用友好,在开源库中使用广泛。
劣
- 效率低,性能差 – 每次java调用so函数时都需要在茫茫jni export函数中寻找函数名一致的,如该功能被频繁调用,延时长性能差;
- 逆向破解危险 – 对应关系明了,在公司内部开发软件时,恐被恶意用户逆向破解。
动态注册
简介
在C代码JNI_OnLoad函数中执行注册。该函数在java使用System.LoadLibrary时执行,即完成java函数与so中函数一一对应的注册。
优
- 性能高 – 相对于静态注册,load库时完成了一一对应的map,每次调用时直接调用该函数而不需要搜寻;
- 防逆向破解更强 – 注册时可以用混淆过的c函数与java函数注册,减少代码可读性。
劣
- 需要程序员手写,没有自动化工具
使用示例
静态注册
STEP1: 写好java中调用native的类和方法
package com.happycoding.lcfan.JniTest;
public class HelloJni {…}
STEP2:javah
命令生成反射类名的头文件.h
在包名目录下,即包含com的文件夹如src,命令行操作如下
javah com.happycoding.lcfan.JniTest.HelloJni
会在该路径下生成 com_happycoding_lcfan_JniTest_HelloJni.h的头文件
STEP3: 根据.h文件在.c文件中写出所列函数的具体实现
实现方法分为Java调C和C调Java,并会根据是否static有不同。具体实现见4。
STEP4: 命令行将C/C++编译成.so/.dll/.jnilib
注意:有可能报错说在LD_LIBRARY_PATH
所列的路径下找不到该文件,那就echo一下,看看.
路径在不在。不在就export。
- mac –
gcc -o LibHelloJniLib.jnilib -lc -shared -I/System/Library/Frameworks/JavaVM.framework/Headers com_happycoding_lcfan_JniTest_HelloJni.c
- linux –
gcc -o libHelloImpl.so -lc -shared -I/usr/local/jdk1.6.0_03/include -I/usr/local/jdk1.6.0_03/include/linux com_happycoding_lcfan_JniTest_HelloJni.c
- 在mac/windows上交叉编译.so – CMake (CMakeLists.txt) or ndk-build (Android.mk + Application.mk)
动态注册
STEP1: 定义 类型为JNINativeMethod 的array:
static JNINativeMethod gMethods[] = {
{"initLogConfig", "(IIII)V", (void *)initLogConfig_jni},
{"initSdkThreadPool", "(Ljava/lang/String;)Z", (void *)initThreadPool_jni},
{"initSdk", "(Lcom/blibee/im/base/sdkjni/IRequestCallback;)V", (void *)initSDK_jni}
}
1-java中的函数名; 2-java函数签名; 3-C中的函数指针
STEP2: 在JNI_OnLoad 中用env->RegisterNatives注册函数
jint JNI_OnLoad(JavaVM *vm, void *reserved) {
LOGD("start JNI_LOAD");
//for c++
if (vm->GetEnv((void **)&env, JNI_VERSION_1_4) != JNI_OK) {
LOGE("getEnv from vm failed.");
return -1;
}
//for c
// if ((*vm)->GetEnv(vm, (void**) &env, JNI_VERSION_1_4) != JNI_OK) {return -1;}
// assert(env != NULL);
jclass clazz = env->FindClass(JNIREG_CLASS);
int nRes = env->RegisterNatives(clazz, gMethods, sizeof(gMethods) / sizeof(gMethods[0]));//注册
if (nRes < 0) {
LOGD("env->RegisterNatives failed.");
return JNI_ERR;
}
LOGD("JNI_OnLoad END!");
return JNI_VERSION_1_4; //必须返回这个值,否则报错
}