android JNI实验心得

编写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”编码的,所以会出现乱码,需要转换编码

猜你喜欢

转载自892848153.iteye.com/blog/1988260