JNI_Onload在执行system.loadLibrary()函数时被调用,主要用途:
1.通过JNI_Onload告知VM,当前so库使用的JNI版本,最老的版本问JNI 1.1(JNI_Onload默认返回的是1.1版本),最新的JNI 1.4;其中JNI 1.4做了很多扩充。
2.可以在JNI_Onload中进行数据的初始化。
3.可以在JNI_Onload对java类中的native函数进行注册。java类是通过VM来调用本地方法,调用时需要通过VM在so库中寻找该本地函数,如果该本地函数需要频繁调用的话,会花费很多时间,可以在JNI_Onload调用registerNativeMethods,把native函数注册到VM中,减少寻找花费的时间。
实践:(1)在eclipse中创建一个android项目,新建一个JniMethod类,主要用来调用native方法。
JniMethod.java
package com.example.testjnionload;
public class JniMethod {
static{
System.loadLibrary("tjnionlod");
}
public native void printLog();
public native void addFunc(int a,int b);
}
(2)在项目下新建一个jni文件夹,通过javah 生成JniMethod类中的native函数的声明头文件,如图1,2。
图1
图2
(3)在jni文件夹中新建一个jnimethod.c源文件,用于实现JNI_Onload和native函数。
jnimethod.c
#include
#include
#include
#define TAG "jnimethod"
#define LOGI(...) __android_log_print(ANDROID_LOG_INFO,TAG,__VA_ARGS__)
#define JNIREG_CLASS "com/example/testjnionload/JniMethod" //native函数的java类名
JNIEXPORT void JNICALL native_printLog(JNIEnv *env, jobject obj){ //输出log信息
LOGI("print my log");
}
JNIEXPORT void JNICALL native_addFunc(JNIEnv *env, jobject obj, jint a, jint b){ //在本地函数中实现加法
int a1,b1,c1;
a1=a;
b1=b;
c1=a1+b1;
LOGI("addFunc return:%d",c1);
}
static JNINativeMethod gMethods[]={{"printLog","()V",(void*)native_printLog},
{"addFunc","(II)V",(void*)native_addFunc} }; //需要注册到VM的native函数-->java中声明的native函数,函数JNI类型,本地对于的native函数名
static int registerNativeMethods(JNIEnv *env){ //native函数的注册
jclass clazz;
LOGI("in registerNativeMethods");
clazz=(*env)->FindClass(env,JNIREG_CLASS);
if(clazz==NULL){
return JNI_FALSE;
}
if((*env)->RegisterNatives(env,clazz,gMethods,sizeof(gMethods)/sizeof(gMethods[0]))<0){ //注册函数
return JNI_FALSE;
}
return JNI_TRUE;
}
JNIEXPORT jint JNICALL JNI_OnLoad(JavaVM* vm,void* reserved){
JNIEnv *env;
jint result=-1;
LOGI("in JNI_Onload");
if((*vm)->GetEnv(vm,(void**)&env,JNI_VERSION_1_4)!=JNI_OK){
return -1;
}
// assert(env!=NULL);
if(!registerNativeMethods(env)){
return -1;
}
result=JNI_VERSION_1_4; //返回JNI 1.4版本信息给VM
return result;
}
编写Android.mk文件
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE :=tjnionlod
LOCAL_SRC_FILES :=jnimethod.c
LOCAL_LDLIBS :=-llog
include $(BUILD_SHARED_LIBRARY)
然后在eclipse中调用ndk-build进行编译,生成对于的so文件。
在android项目的主activity中,调用该native函数,如图3。
图3
JNI_Onload的基本使用就完成了。和JNI_Onload对应的还有JNI_Unload,JNI_Onload在加载so文件时调用的,JNI_Unload在卸载so时调用的。