我们知道一般在jni开发时,需要先根据java文件,利用javah生产对应的.h文件,再实现相应方法。运行时Java虚拟机加载相应运行库,再逐一检索,将java类中的本地方法和相应的JNI本地函数方法映射起来,但当本地接口方法很多时,会增加运行时间,降低运行效率。为了解决这一问题,JNI机制提供了RegisterNatives() JNI函数,该函数允许JNI本地函数与java类中的本地方法直接映射在一起。
使用相应库时,首先需要调用System.loadLibrary()加载相应库,该方法执行过程中,会检查JNI_OnLoad()函数是否被实现,因此想要直接注册本地函数,需要在JNI_OnLoad()中调用RegisterNatives() 函数进行映射匹配。
假设存在一个java类,包含一个native接口
package com.zpm.Info;
public class ZPm {
static{
System.loadLibrary("zpm");
}
public native String get();
}
在jni目录新建一个cpp文件,文件名随便,这里采用的是C++,采用C时,语法略有不同
/*
* Zpm.cpp
*
* Created on: 2018-7-3
* Author: z26442
*/
#include<jni.h>
#include<stdio.h>
jstring hi(JNIEnv* env, jobject obj){
return env->NewStringUTF("zhupumao");
}
const JNINativeMethod method[]={
{"get","()Ljava/lang/String;",reinterpret_cast<jstring*>(hi)}
};
JNIEXPORT jint JNICALL JNI_OnLoad(JavaVM *jvm,void *reserved){
JNIEnv *env;
if(jvm->GetEnv((void**)&env,JNI_VERSION_1_4)!=JNI_OK){
return -1;
}
jclass clz=env->FindClass("com/zpm/Info/ZPm");
env->RegisterNatives(clz,method,sizeof(method)/sizeof(method[0]));
return JNI_VERSION_1_4;
}
新建Android.mk文件
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE := zpm
LOCAL_SRC_FILES := Zpm.cpp
include $(BUILD_SHARED_LIBRARY)
编译生产.so库。这样就可以使用了。