在android中调用C语言接口时,要为native函数名称,命名规则是Java_包名(点用下划线替换)_类名_函数自定义名称,举个例子:

在java类JninameActivity(包名为com.ckl.jniname)中声明native接口:

private native String getJniString();

对应Jni函数名称需定义为下面这样:

jstring Java_com_ckl_jniname_JninameActivity_getJniString(JNIEnv* env, jobject thiz)
{
//
}


如果使用的接口较多,这么长的函数名称将是比较麻烦的事情,这里使用宏定义的方法,可以减轻jni函数命名的工作量。

//定义包名+类名

#define classcom_ckl_jniname_jninameactivity

//这里的宏一定要这样定义,才能合成出正确的函数名称

#define name3(class3, func3) Java_##class3##_##func3

#define name2(class2, func2) name3(class2, func2)

#define name(func) name2(class, func)

//name(getjnistring) 编译时将被替换为 java_com_ckl_jniname_jninameactivity_getjnistring

jstring name(getjnistring)(jnienv* env, jobject thiz)

{

return (*env)->newstringutf(env, "i am form jni !");

}

jstring name(getname)(jnienv* env, jobject thiz)

{

return (*env)->newstringutf(env, "i am jack, ");

}

jstring name(getage)(jnienv* env, jobject thiz)

{

return (*env)->newstringutf(env, "20 !");

}



关键是宏NAME(FUNC)的定义,一定按照下面的方法定义,才能合成出正确的函数名称

#define NAME3(CLASS3, FUNC3) Java_##CLASS3##_##FUNC3
#define NAME2(CLASS2, FUNC2) NAME3(CLASS2, FUNC2)
#define NAME(FUNC) NAME2(CLASS, FUNC)



当然,还有另外一种更常用的方法,

即在JNI_OnLoad()中使用(*env)->RegisterNatives()将需要用到的native接口注册好,

这样就不用按照上面的规则给jni函数取很长的名称了。