JNI函数名称分为三部分:
- 首先是Java关键字,供Java虚拟机识别;例如Java_com_example_hellojni_HelloJni_stringFromJNI
- Java:关键字
- com_example_hellojni:包名
- HelloJni:文件名称(原来的名称为hello_jni)
- stringFromJNI:函数名称
- 然后是调用者类名称(全限定的类名,其中用下划线代替名称分隔符);
- 最后是对应的方法名称,各段名称之间用下划线分割。
JNI函数_的参数也由三部分组成:
- 首先是JNIEnv *,是一个指向JNI运行环境的指针;
- 第二个参数随本地方法是静态还是非静态而有所不同一一非静态本地方法的第二个参数是对对象的引用,而静态本地方法的第二个参数是对其Java类的引用;
- 其余的参数对应通常Java方法的参数,参数类型需要根据一定规则进行映射。
jstring //返回类型,String
Java_com_example_hellojni_HelloJni_stringFromJNI( JNIEnv* env, jobject thiz )
{
return (*env)->NewStringUTF(env, "Hello from JNI !");
}
Java_com_example_hellojni_HelloJni_stringFromJNI( JNIEnv* env, jobject thiz )
{
return (*env)->NewStringUTF(env, "Hello from JNI !");
}
编写make文件:
LOCAL_PATH := $(call my-dir) //源文件地址,必须在第一行include $(CLEAR_VARS) //清楚一些变量,LOCAL_PATH除外LOCAL_MODULE := hello-jni //指定当前编译模块的名称LOCAL_SRC_FILES := hello-jni.c //源文件include $(BUILD_SHARED_LIBRARY) //将当前模块编译为共享链接库,前缀为lib,后缀为.so
最后在Cygwin交叉编译环境中编译该文件他就会在根目录的libs目录下生成
armeabi/libhello-jni.so文件
把该libs目录拷贝到Eclipse中的根目录就可以使用已编写的本地方法了
代码:
public
class
HelloJni
extends
Activity
{
@Override
public
void
onCreate(Bundle savedInstanceState)
{
super
.onCreate(savedInstanceState);
TextView tv =
new
TextView(
this
);
tv.setText(stringFromJNI());
setContentView(tv);
}
public
native
String stringFromJNI();
public
native
String unimplementedStringFromJNI();
static
{
System.loadLibrary(
"hello-jni"
);
}
}