编写JNI最好用C++编写,因为比较简单
java类中,可以定义一个跟java函数一模一样的native函数,不会出现冲突。
JNIEXPORT, JNICALL关键字 没必要加,不加也能运行。
静态注册:根据JNI的函数名来注册
注册规则:函数名必须是 "Java_" + 包名 + 类名 + 方法名,"." 符号一律换成“_”。
函数的前两个参数必须是(JNIEnv *env, jobject object )
供java层调用的函数都必须采用"C"的编译方式编译,所以若是采用.cpp文件,即使用C++编写native 层代码,则要在文件头中适用extern "C" {},在{} 中声明被调用函数
动态注册:动态注册时,函数名没必要按JNI函数名规则来取。
//声明使用"C"的方式编译
extern "C"
{
jstring com_lj_nativetest_NativeApi_helloWorld(JNIEnv *env, jobject object);
}
jstring com_lj_nativetest_NativeApi_helloWorld(JNIEnv *env, jobject object)
{
jstring hello = env->NewStringUTF("Hello from JNI !");
return hello;
}
//这个最好是静态的,因为android的官方文档说:
- In
JNI_OnLoad
, register all of your native methods. You should declare the methods "static" so the names don't take up space in the symbol table on the device.
static JNINativeMethod gMethods[] = {
{
"helloWorld",
"()Ljava/lang/String;",
(void*) com_lj_nativetest_NativeApi_helloWorld
}
};
int register_native_methods(JNIEnv *env, const char* className, const JNINativeMethod* gMethods, int numMethods)
{
jclass clazz;
clazz = env->FindClass(className);
if (env->RegisterNatives(clazz, gMethods, numMethods) < 0)
{
return JNI_FALSE;
}
return JNI_TRUE;
}
jint JNI_Onload(JavaVM* vm, void* reserved)
{
JNIEnv* env;
jint result = -1;
if (vm->GetEnv((void**) env, JNI_VERSION_1_6) != JNI_OK)
{
return -1;
}
if (register_native_methods(env, "com/lj/nativetest/NativeApi", gMethods, sizeof(gMethods) / sizeof(gMethods[0])))
{
goto bail;
}
bail:
return result;
}
- 以上动态注册代码在我的机子上会报错,我也不知道为什么, 包名路径什么的都没错,就是找不到。还有我的Eclipse不能识别C++的“NULL”,这是哪里出问题了,有带查证。
JNI 中的string 默认是"Modified UTF-8"编码的, java语言默认是使用“UTF-16”编码的,所以会出现乱码,需要转换编码